diff --git a/dist/index.js b/dist/index.js deleted file mode 100644 index d802340..0000000 --- a/dist/index.js +++ /dev/null @@ -1,3 +0,0 @@ -/*! For license information please see index.js.LICENSE.txt */ -!function(t,e){"object"==typeof exports&&"object"==typeof module?module.exports=e(require("three"),require("plotly.js-dist"),require("three/examples/jsm/Addons"),require("three/examples/jsm/libs/stats.module")):"function"==typeof define&&define.amd?define(["three","plotly.js-dist","three/examples/jsm/Addons","three/examples/jsm/libs/stats.module"],e):"object"==typeof exports?exports.nbody=e(require("three"),require("plotly.js-dist"),require("three/examples/jsm/Addons"),require("three/examples/jsm/libs/stats.module")):t.nbody=e(t.three,t["plotly.js-dist"],t["three/examples/jsm/Addons"],t["three/examples/jsm/libs/stats.module"])}(this,((t,e,i,s)=>(()=>{"use strict";var n={924:t=>{t.exports=e},604:e=>{e.exports=t},987:t=>{t.exports=i},686:t=>{t.exports=s}},o={};function r(t){var e=o[t];if(void 0!==e)return e.exports;var i=o[t]={exports:{}};return n[t](i,i.exports,r),i.exports}r.n=t=>{var e=t&&t.__esModule?()=>t.default:()=>t;return r.d(e,{a:e}),e},r.d=(t,e)=>{for(var i in e)r.o(e,i)&&!r.o(t,i)&&Object.defineProperty(t,i,{enumerable:!0,get:e[i]})},r.o=(t,e)=>Object.prototype.hasOwnProperty.call(t,e),r.r=t=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})};var l={};return(()=>{r.r(l),r.d(l,{BodyCenterTransformation:()=>P,CelestialBody:()=>t,CentripetalForce:()=>s,CoMTransformation:()=>G,CombinedForce:()=>n,ExplicitEulerSim:()=>h,Gravity:()=>i,LambdaSim:()=>u,LambdaTransformation:()=>Y,RealTimeVisualizer:()=>z,RealTimeVisualizer3D:()=>O,RecordingVisualizer:()=>j,RecordingVisualizer3D:()=>U,RotateTransformation:()=>R,RungeKutta4Sim:()=>c,SemiImplicitEulerSim:()=>d,Simulation:()=>H,State:()=>o,Universe:()=>K,Vector3:()=>e.Vector3,VelocityVerletSim:()=>a});class t{constructor(t,e,i,s,n){this.label=t,this.mass=e,this.position=i,this.velocity=s,this.acceleration=n}clone(e,i,s){return new t(this.label,this.mass,void 0===e?this.position.clone():e,void 0===i?this.velocity.clone():i,void 0===s?this.acceleration.clone():s)}}var e=r(604);class i{constructor(t=6674e-14){this.G=t}getForces(t){let i=t.length,s=[];for(let t=0;t{const e=this.center.clone().sub(t.position);return e.setLength(t.mass*t.velocity.lengthSq()/e.length())}))}}class n{constructor(t){this.forces=t}getForces(t){const i=t.map((()=>new e.Vector3(0,0,0)));return this.forces.forEach((e=>{e.getForces(t).forEach(((t,e)=>{i[e].add(t)}))})),i}}class o{constructor(t){this.bodies=t}clone(){return new o(this.bodies.map((t=>t.clone())))}}class a{constructor(t){this.forceCalculator=t}simulate(t,e){if(t<=0)return e.clone();let i=e.bodies.map((e=>{let i=e.clone();return i.position=this.calcNewPos(i.position,i.velocity,i.acceleration,t),i})),s=this.forceCalculator.getForces(i);return new o(i.map(((e,i)=>{let n=s[i].divideScalar(e.mass);return e.velocity.add(e.acceleration.add(n).multiplyScalar(t/2)),e.acceleration=n,e})))}calcNewPos(t,e,i,s){return t.clone().add(e.clone().multiplyScalar(s)).add(i.clone().multiplyScalar(s*s*.5))}}class h{constructor(t){this.force=t}simulate(t,e){const i=e.bodies.map((e=>e.clone(this.rateUpdate(e.position,e.velocity,t),this.rateUpdate(e.velocity,e.acceleration,t)))),s=this.force.getForces(i);return i.forEach(((t,e)=>{t.acceleration=s[e].divideScalar(t.mass)})),new o(i)}rateUpdate(t,e,i){return e.clone().multiplyScalar(i).add(t)}}class d{constructor(t){this.force=t}simulate(t,e){const i=e.bodies.map((e=>{const i=this.rateUpdate(e.velocity,e.acceleration,t);return e.clone(this.rateUpdate(e.position,i,t),i)})),s=this.force.getForces(i);return i.forEach(((t,e)=>{t.acceleration=s[e].divideScalar(t.mass)})),new o(i)}rateUpdate(t,e,i){return e.clone().multiplyScalar(i).add(t)}}class c{constructor(t,e){if(this.force=t,4!==e.length)throw new Error("Weights for RK4 must be of length 4");this.weights=e}simulate(t,i){let s=i.bodies.map((t=>({kv:[t.acceleration.clone()],kx:[t.velocity.clone()]})));const n=this.getInterKV(i.bodies,s,0,t/2),r=this.getInterKX(i.bodies,s,0,t/2);s.forEach(((t,e)=>{t.kv.push(n[e]),t.kx.push(r[e])}));const l=this.getInterKV(i.bodies,s,1,t/2),a=this.getInterKX(i.bodies,s,1,t/2);s.forEach(((t,e)=>{t.kv.push(l[e]),t.kx.push(a[e])}));const h=this.getInterKV(i.bodies,s,2,t),d=this.getInterKX(i.bodies,s,2,t);s.forEach(((t,e)=>{t.kv.push(h[e]),t.kx.push(d[e])}));const c=i.bodies.map(((i,n)=>{const o=new e.Vector3,r=new e.Vector3;return s[n].kx.forEach(((t,e)=>{o.add(t.multiplyScalar(this.weights[e]))})),s[n].kv.forEach(((t,e)=>{r.add(t.multiplyScalar(this.weights[e]))})),i.clone(o.multiplyScalar(t/6).add(i.position),r.multiplyScalar(t/6).add(i.velocity))})),u=this.force.getForces(c);return c.forEach(((t,e)=>{t.acceleration=u[e].divideScalar(t.mass)})),new o(c)}getInterKV(t,e,i,s){let n=t.map(((t,n)=>{let o=t.clone();return o.position.add(e[n].kx[i].clone().multiplyScalar(s)),o}));return this.force.getForces(n).map(((e,i)=>e.divideScalar(t[i].mass)))}getInterKX(t,e,i,s){return t.map(((t,n)=>t.velocity.clone().add(e[n].kv[i].clone().multiplyScalar(s))))}}class u{constructor(t){this.fn=t}simulate(t,e,i){return this.fn(t,e,i)}}class p{constructor(t,e,i,s,n="div"){this.parent=t,this.object=e,this.property=i,this._disabled=!1,this._hidden=!1,this.initialValue=this.getValue(),this.domElement=document.createElement(n),this.domElement.classList.add("controller"),this.domElement.classList.add(s),this.$name=document.createElement("div"),this.$name.classList.add("name"),p.nextNameID=p.nextNameID||0,this.$name.id="lil-gui-name-"+ ++p.nextNameID,this.$widget=document.createElement("div"),this.$widget.classList.add("widget"),this.$disable=this.$widget,this.domElement.appendChild(this.$name),this.domElement.appendChild(this.$widget),this.domElement.addEventListener("keydown",(t=>t.stopPropagation())),this.domElement.addEventListener("keyup",(t=>t.stopPropagation())),this.parent.children.push(this),this.parent.controllers.push(this),this.parent.$children.appendChild(this.domElement),this._listenCallback=this._listenCallback.bind(this),this.name(i)}name(t){return this._name=t,this.$name.textContent=t,this}onChange(t){return this._onChange=t,this}_callOnChange(){this.parent._callOnChange(this),void 0!==this._onChange&&this._onChange.call(this,this.getValue()),this._changed=!0}onFinishChange(t){return this._onFinishChange=t,this}_callOnFinishChange(){this._changed&&(this.parent._callOnFinishChange(this),void 0!==this._onFinishChange&&this._onFinishChange.call(this,this.getValue())),this._changed=!1}reset(){return this.setValue(this.initialValue),this._callOnFinishChange(),this}enable(t=!0){return this.disable(!t)}disable(t=!0){return t===this._disabled||(this._disabled=t,this.domElement.classList.toggle("disabled",t),this.$disable.toggleAttribute("disabled",t)),this}show(t=!0){return this._hidden=!t,this.domElement.style.display=this._hidden?"none":"",this}hide(){return this.show(!1)}options(t){const e=this.parent.add(this.object,this.property,t);return e.name(this._name),this.destroy(),e}min(t){return this}max(t){return this}step(t){return this}decimals(t){return this}listen(t=!0){return this._listening=t,void 0!==this._listenCallbackID&&(cancelAnimationFrame(this._listenCallbackID),this._listenCallbackID=void 0),this._listening&&this._listenCallback(),this}_listenCallback(){this._listenCallbackID=requestAnimationFrame(this._listenCallback);const t=this.save();t!==this._listenPrevValue&&this.updateDisplay(),this._listenPrevValue=t}getValue(){return this.object[this.property]}setValue(t){return this.getValue()!==t&&(this.object[this.property]=t,this._callOnChange(),this.updateDisplay()),this}updateDisplay(){return this}load(t){return this.setValue(t),this._callOnFinishChange(),this}save(){return this.getValue()}destroy(){this.listen(!1),this.parent.children.splice(this.parent.children.indexOf(this),1),this.parent.controllers.splice(this.parent.controllers.indexOf(this),1),this.parent.$children.removeChild(this.domElement)}}class m extends p{constructor(t,e,i){super(t,e,i,"boolean","label"),this.$input=document.createElement("input"),this.$input.setAttribute("type","checkbox"),this.$input.setAttribute("aria-labelledby",this.$name.id),this.$widget.appendChild(this.$input),this.$input.addEventListener("change",(()=>{this.setValue(this.$input.checked),this._callOnFinishChange()})),this.$disable=this.$input,this.updateDisplay()}updateDisplay(){return this.$input.checked=this.getValue(),this}}function g(t){let e,i;return(e=t.match(/(#|0x)?([a-f0-9]{6})/i))?i=e[2]:(e=t.match(/rgb\(\s*(\d*)\s*,\s*(\d*)\s*,\s*(\d*)\s*\)/))?i=parseInt(e[1]).toString(16).padStart(2,0)+parseInt(e[2]).toString(16).padStart(2,0)+parseInt(e[3]).toString(16).padStart(2,0):(e=t.match(/^#?([a-f0-9])([a-f0-9])([a-f0-9])$/i))&&(i=e[1]+e[1]+e[2]+e[2]+e[3]+e[3]),!!i&&"#"+i}const v={isPrimitive:!0,match:t=>"number"==typeof t,fromHexString:t=>parseInt(t.substring(1),16),toHexString:t=>"#"+t.toString(16).padStart(6,0)},b={isPrimitive:!1,match:t=>Array.isArray(t),fromHexString(t,e,i=1){const s=v.fromHexString(t);e[0]=(s>>16&255)/255*i,e[1]=(s>>8&255)/255*i,e[2]=(255&s)/255*i},toHexString:([t,e,i],s=1)=>v.toHexString(t*(s=255/s)<<16^e*s<<8^i*s)},f={isPrimitive:!1,match:t=>Object(t)===t,fromHexString(t,e,i=1){const s=v.fromHexString(t);e.r=(s>>16&255)/255*i,e.g=(s>>8&255)/255*i,e.b=(255&s)/255*i},toHexString:({r:t,g:e,b:i},s=1)=>v.toHexString(t*(s=255/s)<<16^e*s<<8^i*s)},w=[{isPrimitive:!0,match:t=>"string"==typeof t,fromHexString:g,toHexString:g},v,b,f];class y extends p{constructor(t,e,i,s){var n;super(t,e,i,"color"),this.$input=document.createElement("input"),this.$input.setAttribute("type","color"),this.$input.setAttribute("tabindex",-1),this.$input.setAttribute("aria-labelledby",this.$name.id),this.$text=document.createElement("input"),this.$text.setAttribute("type","text"),this.$text.setAttribute("spellcheck","false"),this.$text.setAttribute("aria-labelledby",this.$name.id),this.$display=document.createElement("div"),this.$display.classList.add("display"),this.$display.appendChild(this.$input),this.$widget.appendChild(this.$display),this.$widget.appendChild(this.$text),this._format=(n=this.initialValue,w.find((t=>t.match(n)))),this._rgbScale=s,this._initialValueHexString=this.save(),this._textFocused=!1,this.$input.addEventListener("input",(()=>{this._setValueFromHexString(this.$input.value)})),this.$input.addEventListener("blur",(()=>{this._callOnFinishChange()})),this.$text.addEventListener("input",(()=>{const t=g(this.$text.value);t&&this._setValueFromHexString(t)})),this.$text.addEventListener("focus",(()=>{this._textFocused=!0,this.$text.select()})),this.$text.addEventListener("blur",(()=>{this._textFocused=!1,this.updateDisplay(),this._callOnFinishChange()})),this.$disable=this.$text,this.updateDisplay()}reset(){return this._setValueFromHexString(this._initialValueHexString),this}_setValueFromHexString(t){if(this._format.isPrimitive){const e=this._format.fromHexString(t);this.setValue(e)}else this._format.fromHexString(t,this.getValue(),this._rgbScale),this._callOnChange(),this.updateDisplay()}save(){return this._format.toHexString(this.getValue(),this._rgbScale)}load(t){return this._setValueFromHexString(t),this._callOnFinishChange(),this}updateDisplay(){return this.$input.value=this._format.toHexString(this.getValue(),this._rgbScale),this._textFocused||(this.$text.value=this.$input.value.substring(1)),this.$display.style.backgroundColor=this.$input.value,this}}class A extends p{constructor(t,e,i){super(t,e,i,"function"),this.$button=document.createElement("button"),this.$button.appendChild(this.$name),this.$widget.appendChild(this.$button),this.$button.addEventListener("click",(t=>{t.preventDefault(),this.getValue().call(this.object),this._callOnChange()})),this.$button.addEventListener("touchstart",(()=>{}),{passive:!0}),this.$disable=this.$button}}class x extends p{constructor(t,e,i,s,n,o){super(t,e,i,"number"),this._initInput(),this.min(s),this.max(n);const r=void 0!==o;this.step(r?o:this._getImplicitStep(),r),this.updateDisplay()}decimals(t){return this._decimals=t,this.updateDisplay(),this}min(t){return this._min=t,this._onUpdateMinMax(),this}max(t){return this._max=t,this._onUpdateMinMax(),this}step(t,e=!0){return this._step=t,this._stepExplicit=e,this}updateDisplay(){const t=this.getValue();if(this._hasSlider){let e=(t-this._min)/(this._max-this._min);e=Math.max(0,Math.min(e,1)),this.$fill.style.width=100*e+"%"}return this._inputFocused||(this.$input.value=void 0===this._decimals?t:t.toFixed(this._decimals)),this}_initInput(){this.$input=document.createElement("input"),this.$input.setAttribute("type","text"),this.$input.setAttribute("aria-labelledby",this.$name.id),window.matchMedia("(pointer: coarse)").matches&&(this.$input.setAttribute("type","number"),this.$input.setAttribute("step","any")),this.$widget.appendChild(this.$input),this.$disable=this.$input;const t=t=>{const e=parseFloat(this.$input.value);isNaN(e)||(this._snapClampSetValue(e+t),this.$input.value=this.getValue())};let e,i,s,n,o,r=!1;const l=t=>{if(r){const s=t.clientX-e,n=t.clientY-i;Math.abs(n)>5?(t.preventDefault(),this.$input.blur(),r=!1,this._setDraggingStyle(!0,"vertical")):Math.abs(s)>5&&a()}if(!r){const e=t.clientY-s;o-=e*this._step*this._arrowKeyMultiplier(t),n+o>this._max?o=this._max-n:n+o{this._setDraggingStyle(!1,"vertical"),this._callOnFinishChange(),window.removeEventListener("mousemove",l),window.removeEventListener("mouseup",a)};this.$input.addEventListener("input",(()=>{let t=parseFloat(this.$input.value);isNaN(t)||(this._stepExplicit&&(t=this._snap(t)),this.setValue(this._clamp(t)))})),this.$input.addEventListener("keydown",(e=>{"Enter"===e.key&&this.$input.blur(),"ArrowUp"===e.code&&(e.preventDefault(),t(this._step*this._arrowKeyMultiplier(e))),"ArrowDown"===e.code&&(e.preventDefault(),t(this._step*this._arrowKeyMultiplier(e)*-1))})),this.$input.addEventListener("wheel",(e=>{this._inputFocused&&(e.preventDefault(),t(this._step*this._normalizeMouseWheel(e)))}),{passive:!1}),this.$input.addEventListener("mousedown",(t=>{e=t.clientX,i=s=t.clientY,r=!0,n=this.getValue(),o=0,window.addEventListener("mousemove",l),window.addEventListener("mouseup",a)})),this.$input.addEventListener("focus",(()=>{this._inputFocused=!0})),this.$input.addEventListener("blur",(()=>{this._inputFocused=!1,this.updateDisplay(),this._callOnFinishChange()}))}_initSlider(){this._hasSlider=!0,this.$slider=document.createElement("div"),this.$slider.classList.add("slider"),this.$fill=document.createElement("div"),this.$fill.classList.add("fill"),this.$slider.appendChild(this.$fill),this.$widget.insertBefore(this.$slider,this.$input),this.domElement.classList.add("hasSlider");const t=t=>{const e=this.$slider.getBoundingClientRect();let i=(s=t,n=e.left,o=e.right,r=this._min,(s-n)/(o-n)*(this._max-r)+r);var s,n,o,r;this._snapClampSetValue(i)},e=e=>{t(e.clientX)},i=()=>{this._callOnFinishChange(),this._setDraggingStyle(!1),window.removeEventListener("mousemove",e),window.removeEventListener("mouseup",i)};let s,n,o=!1;const r=e=>{e.preventDefault(),this._setDraggingStyle(!0),t(e.touches[0].clientX),o=!1},l=e=>{if(o){const t=e.touches[0].clientX-s,i=e.touches[0].clientY-n;Math.abs(t)>Math.abs(i)?r(e):(window.removeEventListener("touchmove",l),window.removeEventListener("touchend",a))}else e.preventDefault(),t(e.touches[0].clientX)},a=()=>{this._callOnFinishChange(),this._setDraggingStyle(!1),window.removeEventListener("touchmove",l),window.removeEventListener("touchend",a)},h=this._callOnFinishChange.bind(this);let d;this.$slider.addEventListener("mousedown",(s=>{this._setDraggingStyle(!0),t(s.clientX),window.addEventListener("mousemove",e),window.addEventListener("mouseup",i)})),this.$slider.addEventListener("touchstart",(t=>{t.touches.length>1||(this._hasScrollBar?(s=t.touches[0].clientX,n=t.touches[0].clientY,o=!0):r(t),window.addEventListener("touchmove",l,{passive:!1}),window.addEventListener("touchend",a))}),{passive:!1}),this.$slider.addEventListener("wheel",(t=>{if(Math.abs(t.deltaX)this._max&&(t=this._max),t}_snapClampSetValue(t){this.setValue(this._clamp(this._snap(t)))}get _hasScrollBar(){const t=this.parent.root.$children;return t.scrollHeight>t.clientHeight}get _hasMin(){return void 0!==this._min}get _hasMax(){return void 0!==this._max}}class E extends p{constructor(t,e,i,s){super(t,e,i,"option"),this.$select=document.createElement("select"),this.$select.setAttribute("aria-labelledby",this.$name.id),this.$display=document.createElement("div"),this.$display.classList.add("display"),this.$select.addEventListener("change",(()=>{this.setValue(this._values[this.$select.selectedIndex]),this._callOnFinishChange()})),this.$select.addEventListener("focus",(()=>{this.$display.classList.add("focus")})),this.$select.addEventListener("blur",(()=>{this.$display.classList.remove("focus")})),this.$widget.appendChild(this.$select),this.$widget.appendChild(this.$display),this.$disable=this.$select,this.options(s)}options(t){return this._values=Array.isArray(t)?t:Object.values(t),this._names=Array.isArray(t)?t:Object.keys(t),this.$select.replaceChildren(),this._names.forEach((t=>{const e=document.createElement("option");e.textContent=t,this.$select.appendChild(e)})),this.updateDisplay(),this}updateDisplay(){const t=this.getValue(),e=this._values.indexOf(t);return this.$select.selectedIndex=e,this.$display.textContent=-1===e?t:this._names[e],this}}class S extends p{constructor(t,e,i){super(t,e,i,"string"),this.$input=document.createElement("input"),this.$input.setAttribute("type","text"),this.$input.setAttribute("spellcheck","false"),this.$input.setAttribute("aria-labelledby",this.$name.id),this.$input.addEventListener("input",(()=>{this.setValue(this.$input.value)})),this.$input.addEventListener("keydown",(t=>{"Enter"===t.code&&this.$input.blur()})),this.$input.addEventListener("blur",(()=>{this._callOnFinishChange()})),this.$widget.appendChild(this.$input),this.$disable=this.$input,this.updateDisplay()}updateDisplay(){return this.$input.value=this.getValue(),this}}let _=!1;class ${constructor({parent:t,autoPlace:e=void 0===t,container:i,width:s,title:n="Controls",closeFolders:o=!1,injectStyles:r=!0,touchStyles:l=!0}={}){if(this.parent=t,this.root=t?t.root:this,this.children=[],this.controllers=[],this.folders=[],this._closed=!1,this._hidden=!1,this.domElement=document.createElement("div"),this.domElement.classList.add("lil-gui"),this.$title=document.createElement("div"),this.$title.classList.add("title"),this.$title.setAttribute("role","button"),this.$title.setAttribute("aria-expanded",!0),this.$title.setAttribute("tabindex",0),this.$title.addEventListener("click",(()=>this.openAnimated(this._closed))),this.$title.addEventListener("keydown",(t=>{"Enter"!==t.code&&"Space"!==t.code||(t.preventDefault(),this.$title.click())})),this.$title.addEventListener("touchstart",(()=>{}),{passive:!0}),this.$children=document.createElement("div"),this.$children.classList.add("children"),this.domElement.appendChild(this.$title),this.domElement.appendChild(this.$children),this.title(n),this.parent)return this.parent.children.push(this),this.parent.folders.push(this),void this.parent.$children.appendChild(this.domElement);this.domElement.classList.add("root"),l&&this.domElement.classList.add("allow-touch-styles"),!_&&r&&(function(t){const e=document.createElement("style");e.innerHTML='.lil-gui {\n font-family: var(--font-family);\n font-size: var(--font-size);\n line-height: 1;\n font-weight: normal;\n font-style: normal;\n text-align: left;\n color: var(--text-color);\n user-select: none;\n -webkit-user-select: none;\n touch-action: manipulation;\n --background-color: #1f1f1f;\n --text-color: #ebebeb;\n --title-background-color: #111111;\n --title-text-color: #ebebeb;\n --widget-color: #424242;\n --hover-color: #4f4f4f;\n --focus-color: #595959;\n --number-color: #2cc9ff;\n --string-color: #a2db3c;\n --font-size: 11px;\n --input-font-size: 11px;\n --font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Arial, sans-serif;\n --font-family-mono: Menlo, Monaco, Consolas, "Droid Sans Mono", monospace;\n --padding: 4px;\n --spacing: 4px;\n --widget-height: 20px;\n --title-height: calc(var(--widget-height) + var(--spacing) * 1.25);\n --name-width: 45%;\n --slider-knob-width: 2px;\n --slider-input-width: 27%;\n --color-input-width: 27%;\n --slider-input-min-width: 45px;\n --color-input-min-width: 45px;\n --folder-indent: 7px;\n --widget-padding: 0 0 0 3px;\n --widget-border-radius: 2px;\n --checkbox-size: calc(0.75 * var(--widget-height));\n --scrollbar-width: 5px;\n}\n.lil-gui, .lil-gui * {\n box-sizing: border-box;\n margin: 0;\n padding: 0;\n}\n.lil-gui.root {\n width: var(--width, 245px);\n display: flex;\n flex-direction: column;\n background: var(--background-color);\n}\n.lil-gui.root > .title {\n background: var(--title-background-color);\n color: var(--title-text-color);\n}\n.lil-gui.root > .children {\n overflow-x: hidden;\n overflow-y: auto;\n}\n.lil-gui.root > .children::-webkit-scrollbar {\n width: var(--scrollbar-width);\n height: var(--scrollbar-width);\n background: var(--background-color);\n}\n.lil-gui.root > .children::-webkit-scrollbar-thumb {\n border-radius: var(--scrollbar-width);\n background: var(--focus-color);\n}\n@media (pointer: coarse) {\n .lil-gui.allow-touch-styles, .lil-gui.allow-touch-styles .lil-gui {\n --widget-height: 28px;\n --padding: 6px;\n --spacing: 6px;\n --font-size: 13px;\n --input-font-size: 16px;\n --folder-indent: 10px;\n --scrollbar-width: 7px;\n --slider-input-min-width: 50px;\n --color-input-min-width: 65px;\n }\n}\n.lil-gui.force-touch-styles, .lil-gui.force-touch-styles .lil-gui {\n --widget-height: 28px;\n --padding: 6px;\n --spacing: 6px;\n --font-size: 13px;\n --input-font-size: 16px;\n --folder-indent: 10px;\n --scrollbar-width: 7px;\n --slider-input-min-width: 50px;\n --color-input-min-width: 65px;\n}\n.lil-gui.autoPlace {\n max-height: 100%;\n position: fixed;\n top: 0;\n right: 15px;\n z-index: 1001;\n}\n\n.lil-gui .controller {\n display: flex;\n align-items: center;\n padding: 0 var(--padding);\n margin: var(--spacing) 0;\n}\n.lil-gui .controller.disabled {\n opacity: 0.5;\n}\n.lil-gui .controller.disabled, .lil-gui .controller.disabled * {\n pointer-events: none !important;\n}\n.lil-gui .controller > .name {\n min-width: var(--name-width);\n flex-shrink: 0;\n white-space: pre;\n padding-right: var(--spacing);\n line-height: var(--widget-height);\n}\n.lil-gui .controller .widget {\n position: relative;\n display: flex;\n align-items: center;\n width: 100%;\n min-height: var(--widget-height);\n}\n.lil-gui .controller.string input {\n color: var(--string-color);\n}\n.lil-gui .controller.boolean {\n cursor: pointer;\n}\n.lil-gui .controller.color .display {\n width: 100%;\n height: var(--widget-height);\n border-radius: var(--widget-border-radius);\n position: relative;\n}\n@media (hover: hover) {\n .lil-gui .controller.color .display:hover:before {\n content: " ";\n display: block;\n position: absolute;\n border-radius: var(--widget-border-radius);\n border: 1px solid #fff9;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n }\n}\n.lil-gui .controller.color input[type=color] {\n opacity: 0;\n width: 100%;\n height: 100%;\n cursor: pointer;\n}\n.lil-gui .controller.color input[type=text] {\n margin-left: var(--spacing);\n font-family: var(--font-family-mono);\n min-width: var(--color-input-min-width);\n width: var(--color-input-width);\n flex-shrink: 0;\n}\n.lil-gui .controller.option select {\n opacity: 0;\n position: absolute;\n width: 100%;\n max-width: 100%;\n}\n.lil-gui .controller.option .display {\n position: relative;\n pointer-events: none;\n border-radius: var(--widget-border-radius);\n height: var(--widget-height);\n line-height: var(--widget-height);\n max-width: 100%;\n overflow: hidden;\n word-break: break-all;\n padding-left: 0.55em;\n padding-right: 1.75em;\n background: var(--widget-color);\n}\n@media (hover: hover) {\n .lil-gui .controller.option .display.focus {\n background: var(--focus-color);\n }\n}\n.lil-gui .controller.option .display.active {\n background: var(--focus-color);\n}\n.lil-gui .controller.option .display:after {\n font-family: "lil-gui";\n content: "↕";\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n padding-right: 0.375em;\n}\n.lil-gui .controller.option .widget,\n.lil-gui .controller.option select {\n cursor: pointer;\n}\n@media (hover: hover) {\n .lil-gui .controller.option .widget:hover .display {\n background: var(--hover-color);\n }\n}\n.lil-gui .controller.number input {\n color: var(--number-color);\n}\n.lil-gui .controller.number.hasSlider input {\n margin-left: var(--spacing);\n width: var(--slider-input-width);\n min-width: var(--slider-input-min-width);\n flex-shrink: 0;\n}\n.lil-gui .controller.number .slider {\n width: 100%;\n height: var(--widget-height);\n background: var(--widget-color);\n border-radius: var(--widget-border-radius);\n padding-right: var(--slider-knob-width);\n overflow: hidden;\n cursor: ew-resize;\n touch-action: pan-y;\n}\n@media (hover: hover) {\n .lil-gui .controller.number .slider:hover {\n background: var(--hover-color);\n }\n}\n.lil-gui .controller.number .slider.active {\n background: var(--focus-color);\n}\n.lil-gui .controller.number .slider.active .fill {\n opacity: 0.95;\n}\n.lil-gui .controller.number .fill {\n height: 100%;\n border-right: var(--slider-knob-width) solid var(--number-color);\n box-sizing: content-box;\n}\n\n.lil-gui-dragging .lil-gui {\n --hover-color: var(--widget-color);\n}\n.lil-gui-dragging * {\n cursor: ew-resize !important;\n}\n\n.lil-gui-dragging.lil-gui-vertical * {\n cursor: ns-resize !important;\n}\n\n.lil-gui .title {\n height: var(--title-height);\n line-height: calc(var(--title-height) - 4px);\n font-weight: 600;\n padding: 0 var(--padding);\n -webkit-tap-highlight-color: transparent;\n cursor: pointer;\n outline: none;\n text-decoration-skip: objects;\n}\n.lil-gui .title:before {\n font-family: "lil-gui";\n content: "▾";\n padding-right: 2px;\n display: inline-block;\n}\n.lil-gui .title:active {\n background: var(--title-background-color);\n opacity: 0.75;\n}\n@media (hover: hover) {\n body:not(.lil-gui-dragging) .lil-gui .title:hover {\n background: var(--title-background-color);\n opacity: 0.85;\n }\n .lil-gui .title:focus {\n text-decoration: underline var(--focus-color);\n }\n}\n.lil-gui.root > .title:focus {\n text-decoration: none !important;\n}\n.lil-gui.closed > .title:before {\n content: "▸";\n}\n.lil-gui.closed > .children {\n transform: translateY(-7px);\n opacity: 0;\n}\n.lil-gui.closed:not(.transition) > .children {\n display: none;\n}\n.lil-gui.transition > .children {\n transition-duration: 300ms;\n transition-property: height, opacity, transform;\n transition-timing-function: cubic-bezier(0.2, 0.6, 0.35, 1);\n overflow: hidden;\n pointer-events: none;\n}\n.lil-gui .children:empty:before {\n content: "Empty";\n padding: 0 var(--padding);\n margin: var(--spacing) 0;\n display: block;\n height: var(--widget-height);\n font-style: italic;\n line-height: var(--widget-height);\n opacity: 0.5;\n}\n.lil-gui.root > .children > .lil-gui > .title {\n border: 0 solid var(--widget-color);\n border-width: 1px 0;\n transition: border-color 300ms;\n}\n.lil-gui.root > .children > .lil-gui.closed > .title {\n border-bottom-color: transparent;\n}\n.lil-gui + .controller {\n border-top: 1px solid var(--widget-color);\n margin-top: 0;\n padding-top: var(--spacing);\n}\n.lil-gui .lil-gui .lil-gui > .title {\n border: none;\n}\n.lil-gui .lil-gui .lil-gui > .children {\n border: none;\n margin-left: var(--folder-indent);\n border-left: 2px solid var(--widget-color);\n}\n.lil-gui .lil-gui .controller {\n border: none;\n}\n\n.lil-gui label, .lil-gui input, .lil-gui button {\n -webkit-tap-highlight-color: transparent;\n}\n.lil-gui input {\n border: 0;\n outline: none;\n font-family: var(--font-family);\n font-size: var(--input-font-size);\n border-radius: var(--widget-border-radius);\n height: var(--widget-height);\n background: var(--widget-color);\n color: var(--text-color);\n width: 100%;\n}\n@media (hover: hover) {\n .lil-gui input:hover {\n background: var(--hover-color);\n }\n .lil-gui input:active {\n background: var(--focus-color);\n }\n}\n.lil-gui input:disabled {\n opacity: 1;\n}\n.lil-gui input[type=text],\n.lil-gui input[type=number] {\n padding: var(--widget-padding);\n -moz-appearance: textfield;\n}\n.lil-gui input[type=text]:focus,\n.lil-gui input[type=number]:focus {\n background: var(--focus-color);\n}\n.lil-gui input[type=checkbox] {\n appearance: none;\n width: var(--checkbox-size);\n height: var(--checkbox-size);\n border-radius: var(--widget-border-radius);\n text-align: center;\n cursor: pointer;\n}\n.lil-gui input[type=checkbox]:checked:before {\n font-family: "lil-gui";\n content: "✓";\n font-size: var(--checkbox-size);\n line-height: var(--checkbox-size);\n}\n@media (hover: hover) {\n .lil-gui input[type=checkbox]:focus {\n box-shadow: inset 0 0 0 1px var(--focus-color);\n }\n}\n.lil-gui button {\n outline: none;\n cursor: pointer;\n font-family: var(--font-family);\n font-size: var(--font-size);\n color: var(--text-color);\n width: 100%;\n height: var(--widget-height);\n text-transform: none;\n background: var(--widget-color);\n border-radius: var(--widget-border-radius);\n border: none;\n}\n@media (hover: hover) {\n .lil-gui button:hover {\n background: var(--hover-color);\n }\n .lil-gui button:focus {\n box-shadow: inset 0 0 0 1px var(--focus-color);\n }\n}\n.lil-gui button:active {\n background: var(--focus-color);\n}\n\n@font-face {\n font-family: "lil-gui";\n src: url("data:application/font-woff;charset=utf-8;base64,d09GRgABAAAAAAUsAAsAAAAACJwAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAABHU1VCAAABCAAAAH4AAADAImwmYE9TLzIAAAGIAAAAPwAAAGBKqH5SY21hcAAAAcgAAAD0AAACrukyyJBnbHlmAAACvAAAAF8AAACEIZpWH2hlYWQAAAMcAAAAJwAAADZfcj2zaGhlYQAAA0QAAAAYAAAAJAC5AHhobXR4AAADXAAAABAAAABMAZAAAGxvY2EAAANsAAAAFAAAACgCEgIybWF4cAAAA4AAAAAeAAAAIAEfABJuYW1lAAADoAAAASIAAAIK9SUU/XBvc3QAAATEAAAAZgAAAJCTcMc2eJxVjbEOgjAURU+hFRBK1dGRL+ALnAiToyMLEzFpnPz/eAshwSa97517c/MwwJmeB9kwPl+0cf5+uGPZXsqPu4nvZabcSZldZ6kfyWnomFY/eScKqZNWupKJO6kXN3K9uCVoL7iInPr1X5baXs3tjuMqCtzEuagm/AAlzQgPAAB4nGNgYRBlnMDAysDAYM/gBiT5oLQBAwuDJAMDEwMrMwNWEJDmmsJwgCFeXZghBcjlZMgFCzOiKOIFAB71Bb8AeJy1kjFuwkAQRZ+DwRAwBtNQRUGKQ8OdKCAWUhAgKLhIuAsVSpWz5Bbkj3dEgYiUIszqWdpZe+Z7/wB1oCYmIoboiwiLT2WjKl/jscrHfGg/pKdMkyklC5Zs2LEfHYpjcRoPzme9MWWmk3dWbK9ObkWkikOetJ554fWyoEsmdSlt+uR0pCJR34b6t/TVg1SY3sYvdf8vuiKrpyaDXDISiegp17p7579Gp3p++y7HPAiY9pmTibljrr85qSidtlg4+l25GLCaS8e6rRxNBmsnERunKbaOObRz7N72ju5vdAjYpBXHgJylOAVsMseDAPEP8LYoUHicY2BiAAEfhiAGJgZWBgZ7RnFRdnVJELCQlBSRlATJMoLV2DK4glSYs6ubq5vbKrJLSbGrgEmovDuDJVhe3VzcXFwNLCOILB/C4IuQ1xTn5FPilBTj5FPmBAB4WwoqAHicY2BkYGAA4sk1sR/j+W2+MnAzpDBgAyEMQUCSg4EJxAEAwUgFHgB4nGNgZGBgSGFggJMhDIwMqEAYAByHATJ4nGNgAIIUNEwmAABl3AGReJxjYAACIQYlBiMGJ3wQAEcQBEV4nGNgZGBgEGZgY2BiAAEQyQWEDAz/wXwGAAsPATIAAHicXdBNSsNAHAXwl35iA0UQXYnMShfS9GPZA7T7LgIu03SSpkwzYTIt1BN4Ak/gKTyAeCxfw39jZkjymzcvAwmAW/wgwHUEGDb36+jQQ3GXGot79L24jxCP4gHzF/EIr4jEIe7wxhOC3g2TMYy4Q7+Lu/SHuEd/ivt4wJd4wPxbPEKMX3GI5+DJFGaSn4qNzk8mcbKSR6xdXdhSzaOZJGtdapd4vVPbi6rP+cL7TGXOHtXKll4bY1Xl7EGnPtp7Xy2n00zyKLVHfkHBa4IcJ2oD3cgggWvt/V/FbDrUlEUJhTn/0azVWbNTNr0Ens8de1tceK9xZmfB1CPjOmPH4kitmvOubcNpmVTN3oFJyjzCvnmrwhJTzqzVj9jiSX911FjeAAB4nG3HMRKCMBBA0f0giiKi4DU8k0V2GWbIZDOh4PoWWvq6J5V8If9NVNQcaDhyouXMhY4rPTcG7jwYmXhKq8Wz+p762aNaeYXom2n3m2dLTVgsrCgFJ7OTmIkYbwIbC6vIB7WmFfAAAA==") format("woff");\n}';const i=document.querySelector("head link[rel=stylesheet], head style");i?document.head.insertBefore(e,i):document.head.appendChild(e)}(),_=!0),i?i.appendChild(this.domElement):e&&(this.domElement.classList.add("autoPlace"),document.body.appendChild(this.domElement)),s&&this.domElement.style.setProperty("--width",s+"px"),this._closeFolders=o}add(t,e,i,s,n){if(Object(i)===i)return new E(this,t,e,i);const o=t[e];switch(typeof o){case"number":return new x(this,t,e,i,s,n);case"boolean":return new m(this,t,e);case"string":return new S(this,t,e);case"function":return new A(this,t,e)}console.error("gui.add failed\n\tproperty:",e,"\n\tobject:",t,"\n\tvalue:",o)}addColor(t,e,i=1){return new y(this,t,e,i)}addFolder(t){const e=new $({parent:this,title:t});return this.root._closeFolders&&e.close(),e}load(t,e=!0){return t.controllers&&this.controllers.forEach((e=>{e instanceof A||e._name in t.controllers&&e.load(t.controllers[e._name])})),e&&t.folders&&this.folders.forEach((e=>{e._title in t.folders&&e.load(t.folders[e._title])})),this}save(t=!0){const e={controllers:{},folders:{}};return this.controllers.forEach((t=>{if(!(t instanceof A)){if(t._name in e.controllers)throw new Error(`Cannot save GUI with duplicate property "${t._name}"`);e.controllers[t._name]=t.save()}})),t&&this.folders.forEach((t=>{if(t._title in e.folders)throw new Error(`Cannot save GUI with duplicate folder "${t._title}"`);e.folders[t._title]=t.save()})),e}open(t=!0){return this._setClosed(!t),this.$title.setAttribute("aria-expanded",!this._closed),this.domElement.classList.toggle("closed",this._closed),this}close(){return this.open(!1)}_setClosed(t){this._closed!==t&&(this._closed=t,this._callOnOpenClose(this))}show(t=!0){return this._hidden=!t,this.domElement.style.display=this._hidden?"none":"",this}hide(){return this.show(!1)}openAnimated(t=!0){return this._setClosed(!t),this.$title.setAttribute("aria-expanded",!this._closed),requestAnimationFrame((()=>{const e=this.$children.clientHeight;this.$children.style.height=e+"px",this.domElement.classList.add("transition");const i=t=>{t.target===this.$children&&(this.$children.style.height="",this.domElement.classList.remove("transition"),this.$children.removeEventListener("transitionend",i))};this.$children.addEventListener("transitionend",i);const s=t?this.$children.scrollHeight:0;this.domElement.classList.toggle("closed",!t),requestAnimationFrame((()=>{this.$children.style.height=s+"px"}))})),this}title(t){return this._title=t,this.$title.textContent=t,this}reset(t=!0){return(t?this.controllersRecursive():this.controllers).forEach((t=>t.reset())),this}onChange(t){return this._onChange=t,this}_callOnChange(t){this.parent&&this.parent._callOnChange(t),void 0!==this._onChange&&this._onChange.call(this,{object:t.object,property:t.property,value:t.getValue(),controller:t})}onFinishChange(t){return this._onFinishChange=t,this}_callOnFinishChange(t){this.parent&&this.parent._callOnFinishChange(t),void 0!==this._onFinishChange&&this._onFinishChange.call(this,{object:t.object,property:t.property,value:t.getValue(),controller:t})}onOpenClose(t){return this._onOpenClose=t,this}_callOnOpenClose(t){this.parent&&this.parent._callOnOpenClose(t),void 0!==this._onOpenClose&&this._onOpenClose.call(this,t)}destroy(){this.parent&&(this.parent.children.splice(this.parent.children.indexOf(this),1),this.parent.folders.splice(this.parent.folders.indexOf(this),1)),this.domElement.parentElement&&this.domElement.parentElement.removeChild(this.domElement),Array.from(this.children).forEach((t=>t.destroy()))}controllersRecursive(){let t=Array.from(this.controllers);return this.folders.forEach((e=>{t=t.concat(e.controllersRecursive())})),t}foldersRecursive(){let t=Array.from(this.folders);return this.folders.forEach((e=>{t=t.concat(e.foldersRecursive())})),t}}const C=$;var L=r(924),T=r.n(L),k=r(987),F=r(686),M=r.n(F);let I=null;function D(t,e,i){return ti?i:t}class V{constructor(t,e){this.data={x:[],y:[],mode:"markers",marker:{size:1,color:"white"}},this.data.marker.color=e,this.trailLength=0,this.maxTrailLength=t,this.trailInd=0}addTrail(t,e){this.trailLength{!1===t&&this.universeTrails.forEach((t=>t.popAllTrails())),i.showTrails=t}));const s=e.addFolder("Show Universe");s.open(!1),this.simulation.universes.forEach(((t,e)=>{s.add(i.showUniverse,t.label).onChange((s=>{!1===s&&this.universeTrails[e].popAllTrails(),i.showUniverse[t.label]=s}))}))}start(t,e,i){if(""!==this.divId)return void console.error("Simulation already playing. Stop the current playtime before initiating a new one.");this.divId=t;let s=document.getElementById(t);if(null===s)return;let n=0,o=0;this.simulation.universes.forEach((t=>t.currState.bodies.forEach((t=>{n=Math.max(n,Math.abs(t.position.x)),o=Math.max(o,Math.abs(t.position.y))}))));const r=.5*Math.min(i/o,e/n),l={paper_bgcolor:"#000000",plot_bgcolor:"#000000",font:{color:"#bfbfbf"},xaxis:{autorange:!1,range:[-e/2/r,e/2/r]},yaxis:{autorange:!1,range:[-i/2/r,i/2/r]},showlegend:!1};let a;"ui"===this.simulation.controller&&this.addControls(s),this.simulation.showDebugInfo&&(a=new(M()),a.dom.style.position="absolute",a.dom.style.bottom="0px",a.dom.style.removeProperty("top"),s.appendChild(a.dom));const h=this.simulation.universes.flatMap((t=>{const e=new V(this.simulation.getMaxTrailLength(),"string"==typeof t.color?t.color:t.color[0]);this.universeTrails.push(e);const i={x:t.currState.bodies.map((t=>t.position.x)),y:t.currState.bodies.map((t=>t.position.y)),type:"scatter",mode:"markers",marker:{color:t.color,sizemin:6,size:t.currState.bodies.map((t=>Math.min(10,t.mass)))}};return this.simulation.getShowTrails()?(t.currState.bodies.forEach((t=>{e.addTrail(t.position.x,t.position.y)})),[i,e.data]):[i,{x:[],y:[]}]}));T().newPlot(t,h,l,{scrollZoom:!0,modeBarButtonsToRemove:["lasso2d","select2d","toImage","resetScale2d"]});const d=1e3/this.simulation.maxFrameRate;if(null!==I)return;let c=0,u=0;const p=t=>{this.simulation.simulateStep(this.simulation.controls.speed*Math.min(t-u,33.33)/1e3),u=t},m=e=>{if(0===this.simulation.controls.speed||this.simulation.controls.paused)return void(I=requestAnimationFrame(m));if(p(e),d>0&&e-c{if(!this.simulation.getShowUniverse(t.label))return[{x:[],y:[]},{}];const i={x:t.currState.bodies.map((t=>t.position.x)),y:t.currState.bodies.map((t=>t.position.y)),hovertext:t.currState.bodies.map((t=>t.label)),marker:{size:t.currState.bodies.map((t=>Math.min(10,t.mass))),color:t.color,sizemin:6},mode:"markers"};let s={};if(this.simulation.getShowTrails()){const i=this.universeTrails[e];t.currState.bodies.forEach((t=>{i.addTrail(t.position.x,t.position.y)})),s=i.data}return[i,s]}));T().react(t,i,l),this.simulation.showDebugInfo&&a&&a.update(),I=requestAnimationFrame(m)};I=requestAnimationFrame(m)}stop(){T().purge(this.divId),this.divId="",this.universeTrails.forEach((t=>{t.popAllTrails()})),this.universeTrails=[]}}class B{constructor(t,i,s,n){const o=new e.BufferGeometry;o.setAttribute("position",new e.BufferAttribute(new Float32Array(0),3)),this.trails=new e.Points(o,new e.PointsMaterial({color:i,size:.005*n})),s.add(this.trails),this.trailInd=0,this.trailLength=0,this.maxTrailLength=t}addTrail(t){if(this.trailLength{!1===t&&this.universeTrails.forEach((t=>{t.popAllTrails()})),i.showTrails=t}));const s=e.addFolder("Show Universe");s.open(!1),this.simulation.universes.forEach(((t,e)=>{s.add(i.showUniverse,t.label).onChange((s=>{!1===s&&this.universeTrails[e].popAllTrails(),i.showUniverse[t.label]=s}))}))}start(t,i,s){if(void 0!==this.scene)return void console.error("Simulation already playing. Stop the current playtime before initiating a new one.");let n=document.getElementById(t);if(null===n)return;n.style.position="relative";let o=0,r=0;this.simulation.universes.forEach((t=>t.currState.bodies.forEach((t=>{o=Math.max(o,Math.abs(t.position.x)),r=Math.max(r,Math.abs(t.position.y))}))));const l=.5*Math.min(s/r,i/o);this.scene=new e.Scene;const a=new e.OrthographicCamera(i/-2,i/2,s/2,s/-2,0,1e10);a.position.set(0,0,Math.max(i,s));const h=new e.WebGLRenderer;let d;h.setSize(i,s),h.autoClear=!1,n.appendChild(h.domElement),this.simulation.showDebugInfo&&(d=new(M()),d.dom.style.position="absolute",d.dom.style.right="0px",d.dom.style.removeProperty("left"),n.appendChild(d.dom)),"ui"===this.simulation.controller&&this.addControls(n);const c=new k.OrbitControls(a,h.domElement);c.listenToKeyEvents(window),c.update();const u=new e.AxesHelper(i);this.scene.add(u);const p=new k.ViewHelper(a,h.domElement);let m=[];this.simulation.universes.forEach((t=>{this.universeTrails.push(new B(this.simulation.maxTrailLength,"string"==typeof t.color?t.color:t.color[0],this.scene,l)),t.currState.bodies.forEach((i=>{const s=new e.SphereGeometry(D(Math.log2(i.mass)-70,10,40),8,8),n=new e.WireframeGeometry(s),o=new e.LineSegments(n,new e.LineBasicMaterial({color:new e.Color(t.color)}));this.scene.add(o),o.position.copy(i.position.clone().multiplyScalar(l)),m.push(o)}))}));const g=1e3/this.simulation.maxFrameRate;let v=performance.now(),b=performance.now();const f=t=>{this.simulation.simulateStep(this.simulation.controls.speed*Math.min(t-v,16.67)/1e3),v=t},w=t=>{if(0===this.simulation.controls.speed||this.simulation.controls.paused)return requestAnimationFrame(w),h.clear(),h.render(this.scene,a),p.render(h),void c.update();if(f(t),g>0&&t-b{this.simulation.controls.showUniverse[t.label]?t.currState.bodies.forEach((t=>{m[e].visible=!0,m[e].position.copy(t.position.clone().multiplyScalar(l)),this.simulation.controls.showTrails&&this.universeTrails[i].addTrail(m[e].position),e++})):t.currState.bodies.forEach((t=>{m[e].visible=!1,e++}))})),requestAnimationFrame(w),h.clear(),h.render(this.scene,a),p.render(h),c.update()};requestAnimationFrame(w)}stop(){var t;null===(t=this.scene)||void 0===t||t.clear(),this.scene=void 0,this.universeTrails.forEach((t=>{t.popAllTrails()})),this.universeTrails=[]}}class j{constructor(t){this.divId="",this.universeTrails=[],this.simulation=t}addControls(t){const e=new C({container:t});e.domElement.style.position="absolute",e.domElement.style.top="0",e.domElement.style.left="0",e.domElement.style.zIndex="1000";const i=this.simulation.controls;e.add(i,"speed"),e.add(i,"showTrails").onChange((t=>{!1===t&&this.universeTrails.forEach((t=>t.popAllTrails())),i.showTrails=t}));const s=e.addFolder("Show Universe");s.open(!1),this.simulation.universes.forEach(((t,e)=>{s.add(i.showUniverse,t.label).onChange((s=>{!1===s&&this.universeTrails[e].popAllTrails(),i.showUniverse[t.label]=s}))}))}start(t,e,i,s){if(""!==this.divId)return void console.error("Simulation already playing. Stop the current playtime before initiating a new one.");this.divId=t;let n=document.getElementById(t);if(null===n)return;let o=0,r=0;this.simulation.universes.forEach((t=>t.currState.bodies.forEach((t=>{o=Math.max(o,Math.abs(t.position.x)),r=Math.max(r,Math.abs(t.position.y))}))));const l=.5*Math.min(i/r,e/o),a=[],h=this.simulation.maxFrameRate*s;let d=1;this.simulation.universes.forEach((t=>{a.push([t.currState.clone()])}));for(let t=0;t{a[e].push(t.currState.clone())}));const c={paper_bgcolor:"#000000",plot_bgcolor:"#000000",font:{color:"#bfbfbf"},xaxis:{autorange:!1,range:[-e/2/l,e/2/l]},yaxis:{autorange:!1,range:[-i/2/l,i/2/l]},showlegend:!1};let u;"ui"===this.simulation.controller&&this.addControls(n),this.simulation.showDebugInfo&&(u=new(M()),u.dom.style.position="absolute",u.dom.style.bottom="0px",u.dom.style.removeProperty("top"),n.appendChild(u.dom));const p=this.simulation.universes.flatMap((t=>{const e=new V(this.simulation.getMaxTrailLength(),"string"==typeof t.color?t.color:t.color[0]);this.universeTrails.push(e);const i={x:t.currState.bodies.map((t=>t.position.x)),y:t.currState.bodies.map((t=>t.position.y)),type:"scatter",mode:"markers",marker:{color:t.color,sizemin:6,size:t.currState.bodies.map((t=>Math.min(10,t.mass)))}};return this.simulation.getShowTrails()?(t.currState.bodies.forEach((t=>{e.addTrail(t.position.x,t.position.y)})),[i,e.data]):[i,{x:[],y:[]}]}));if(T().newPlot(t,p,c,{scrollZoom:!0,modeBarButtonsToRemove:["zoom2d","lasso2d","select2d","toImage","resetScale2d"]}),null!==I)return;const m=e=>{if(0===this.simulation.controls.speed||this.simulation.controls.paused)return void(I=requestAnimationFrame(m));const i=Math.round(d),s=this.simulation.universes.flatMap(((t,e)=>{if(!this.simulation.getShowUniverse(t.label))return[{x:[],y:[]},{}];const s=a[e][i],n={x:s.bodies.map((t=>t.position.x)),y:s.bodies.map((t=>t.position.y)),hovertext:s.bodies.map((t=>t.label)),marker:{size:s.bodies.map((t=>Math.min(10,t.mass))),color:t.color,sizemin:6},mode:"markers"};let o={};if(this.simulation.getShowTrails()){const t=this.universeTrails[e];s.bodies.forEach((e=>{t.addTrail(e.position.x,e.position.y)})),o=t.data}return[n,o]}));T().react(t,s,c),this.simulation.showDebugInfo&&u&&u.update(),d=Math.round(d+this.simulation.controls.speed),d<0?d=this.simulation.looped?(d%h+h)%h:0:d>=h&&(this.simulation.looped?d%=h:d=h-1),I=requestAnimationFrame(m)};I=requestAnimationFrame(m)}stop(){T().purge(this.divId),this.divId="",this.universeTrails=[]}}class U{constructor(t){this.universeTrails=[],this.simulation=t}addControls(t){const e=new C({container:t});e.domElement.style.position="absolute",e.domElement.style.top="0",e.domElement.style.left="0",e.domElement.style.zIndex="1000";const i=this.simulation.controls;e.add(i,"speed"),e.add(i,"showTrails").onChange((t=>{!1===t&&this.universeTrails.forEach((t=>{t.popAllTrails()})),i.showTrails=t}));const s=e.addFolder("Show Universe");s.open(!1),this.simulation.universes.forEach(((t,e)=>{s.add(i.showUniverse,t.label).onChange((s=>{!1===s&&this.universeTrails[e].popAllTrails(),i.showUniverse[t.label]=s}))}))}start(t,i,s,n){if(void 0!==this.scene)return void console.error("Simulation already playing. Stop the current playtime before initiating a new one.");let o=document.getElementById(t);if(null===o)return;let r=0,l=0;this.simulation.universes.forEach((t=>t.currState.bodies.forEach((t=>{r=Math.max(r,Math.abs(t.position.x)),l=Math.max(l,Math.abs(t.position.y))}))));const a=.5*Math.min(s/l,i/r);this.scene=new e.Scene;const h=new e.OrthographicCamera(i/-2,i/2,s/2,s/-2,0,1e10);h.position.set(0,0,Math.max(i,s));const d=new e.WebGLRenderer;let c;d.setSize(i,s),d.autoClear=!1,o.appendChild(d.domElement),this.simulation.showDebugInfo&&(c=new(M()),c.dom.style.position="absolute",c.dom.style.right="0px",c.dom.style.removeProperty("left"),o.appendChild(c.dom)),"ui"===this.simulation.controller&&this.addControls(o);const u=new k.OrbitControls(h,d.domElement);u.listenToKeyEvents(window),u.update();const p=new e.AxesHelper(i);this.scene.add(p);const m=new k.ViewHelper(h,d.domElement);let g=[];this.simulation.universes.forEach((t=>{this.universeTrails.push(new B(this.simulation.maxTrailLength,"string"==typeof t.color?t.color:t.color[0],this.scene,a)),t.currState.bodies.forEach((i=>{const s=new e.SphereGeometry(D(Math.log2(i.mass)-70,10,40),8,8),n=new e.WireframeGeometry(s),o=new e.LineSegments(n,new e.LineBasicMaterial({color:new e.Color(t.color)}));this.scene.add(o),o.position.copy(i.position.clone().multiplyScalar(a)),g.push(o)}))}));const v=[],b=this.simulation.maxFrameRate*n;let f=1;this.simulation.universes.forEach((t=>{v.push([t.currState.clone()])}));for(let t=0;t{v[e].push(t.currState.clone())}));const w=t=>{if(0===this.simulation.controls.speed||this.simulation.controls.paused)return requestAnimationFrame(w),d.clear(),d.render(this.scene,h),m.render(d),void u.update();let e=0;this.simulation.universes.forEach(((t,i)=>{this.simulation.controls.showUniverse[t.label]?v[i][f].bodies.forEach((t=>{g[e].visible=!0,g[e].position.copy(t.position.clone().multiplyScalar(a)),this.simulation.controls.showTrails&&this.universeTrails[i].addTrail(g[e].position),e++})):t.currState.bodies.forEach((()=>{g[e].visible=!1,e++}))})),this.simulation.showDebugInfo&&c&&c.update(),f=Math.round(f+this.simulation.controls.speed),f<0?f=this.simulation.looped?(f%b+b)%b:0:f>=b&&(this.simulation.looped?f%=b:f=b-1),requestAnimationFrame(w),d.clear(),d.render(this.scene,h),m.render(d),u.update()};requestAnimationFrame(w)}stop(){var t;null===(t=this.scene)||void 0===t||t.clear(),this.scene=void 0,this.universeTrails.forEach((t=>{t.popAllTrails()})),this.universeTrails=[]}}class H{constructor(t,{visType:e="2D",record:i=!1,looped:s=!0,controller:n="none",showTrails:o=!1,showDebugInfo:r=!1,maxFrameRate:l=-1,maxTrailLength:a=100}){if(this.controls={speed:1,paused:!0,showTrails:!1,showUniverse:{}},this.universes=Array.isArray(t)?t:[t],this.universes.length>10)throw new Error("Too many universes");if(new Set(this.universes.map((t=>t.label))).size!==this.universes.length)throw new Error("Duplicate label in universes");this.controller=n,this.universes.forEach((t=>{this.controls.showUniverse[t.label]=!0})),this.controls.showTrails=o,this.showDebugInfo=r,this.maxFrameRate=l,this.maxTrailLength=a,this.looped=s,i?(this.maxFrameRate=60,this.visualizer="2D"===e?new j(this):new U(this)):this.visualizer="2D"===e?new z(this):new O(this)}getSpeed(){return this.controls.speed}setSpeed(t){"code"===this.controller&&(this.controls.speed=t)}isPlaying(){return!this.controls.paused}pause(){"code"===this.controller&&(this.controls.paused=!0)}resume(){"code"===this.controller&&(this.controls.paused=!1)}getShowTrails(){return this.controls.showTrails}setShowTrails(t){"code"===this.controller&&(this.controls.showTrails=t)}getShowUniverse(t){return this.controls.showUniverse[t]}setShowUniverse(t,e){"code"===this.controller&&(this.controls.showUniverse[t]=e)}getMaxTrailLength(){return this.maxTrailLength}setMaxTrailLength(t){"code"===this.controller&&(this.maxTrailLength=t)}simulateStep(t){this.universes.forEach((e=>{e.simulateStep(t)}))}start(t,e,i,s=1,n=!1,o=0){if(void 0===o)throw new Error("recordFor must be defined if record is true");this.controls.paused=n,this.controls.speed=s,this.visualizer.start(t,e,i,o)}stop(){this.visualizer.stop()}}class P{transform(t){const e=t.bodies[0].position.clone();return t.bodies.forEach((t=>{t.position.sub(e)})),t}}class G{transform(t){let i=0,s=new e.Vector3;return t.bodies.forEach((t=>{i+=t.mass,s.add(t.position.clone().multiplyScalar(t.mass))})),s.divideScalar(i),t.bodies.forEach((t=>{t.position.sub(s)})),t}}class R{constructor(t,e){this.axis=t,this.angle=e}transform(t){return t.bodies.forEach((t=>{t.position.applyAxisAngle(this.axis,this.angle),t.velocity.applyAxisAngle(this.axis,this.angle),t.acceleration.applyAxisAngle(this.axis,this.angle)})),t}}class Y{constructor(t){this.fn=t}transform(t,e){return this.fn(t,e)}}class K{constructor(t){if(void 0===t.currState)throw new Error("Missing Current State in Universe");if(void 0===t.simFunc)throw new Error("Missing Simulation Function in Universe");this.label=void 0===t.label?"Universe":t.label,this.prevState=void 0===t.prevState?t.currState:t.prevState,this.currState=t.currState,this.color=void 0===t.color?"rgba(255, 255, 255, 1)":t.color,this.simFunc=t.simFunc,this.transformations=void 0===t.transformations?[]:Array.isArray(t.transformations)?t.transformations:[t.transformations]}simulateStep(t){let e=this.simFunc.simulate(t,this.currState,this.prevState);this.prevState=this.currState,this.transformations.forEach((i=>{e=i.transform(e,t)})),this.currState=e}clone(){return new K({prevState:this.prevState.clone(),currState:this.currState.clone(),color:this.color,label:this.label,simFunc:this.simFunc,transformations:this.transformations})}}})(),l})())); -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.js","mappings":"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","sources":["webpack://nbody/webpack/universalModuleDefinition","webpack://nbody/external umd \"plotly.js-dist\"","webpack://nbody/external umd \"three\"","webpack://nbody/external umd \"three/examples/jsm/Addons\"","webpack://nbody/external umd \"three/examples/jsm/libs/stats.module\"","webpack://nbody/webpack/bootstrap","webpack://nbody/webpack/runtime/compat get default export","webpack://nbody/webpack/runtime/define property getters","webpack://nbody/webpack/runtime/hasOwnProperty shorthand","webpack://nbody/webpack/runtime/make namespace object","webpack://nbody/./src/CelestialBody.ts","webpack://nbody/./src/library/Force.ts","webpack://nbody/./src/State.ts","webpack://nbody/./src/library/SimulateFunction.ts","webpack://nbody/./src/SimulateFunction.ts","webpack://nbody/./node_modules/lil-gui/dist/lil-gui.esm.js","webpack://nbody/./src/library/Visualizer.ts","webpack://nbody/./src/Simulation.ts","webpack://nbody/./src/library/Transformation.ts","webpack://nbody/./src/Transformation.ts","webpack://nbody/./src/Universe.ts"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory(require(\"three\"), require(\"plotly.js-dist\"), require(\"three/examples/jsm/Addons\"), require(\"three/examples/jsm/libs/stats.module\"));\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([\"three\", \"plotly.js-dist\", \"three/examples/jsm/Addons\", \"three/examples/jsm/libs/stats.module\"], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"nbody\"] = factory(require(\"three\"), require(\"plotly.js-dist\"), require(\"three/examples/jsm/Addons\"), require(\"three/examples/jsm/libs/stats.module\"));\n\telse\n\t\troot[\"nbody\"] = factory(root[\"three\"], root[\"plotly.js-dist\"], root[\"three/examples/jsm/Addons\"], root[\"three/examples/jsm/libs/stats.module\"]);\n})(this, (__WEBPACK_EXTERNAL_MODULE__604__, __WEBPACK_EXTERNAL_MODULE__924__, __WEBPACK_EXTERNAL_MODULE__987__, __WEBPACK_EXTERNAL_MODULE__686__) => {\nreturn ","module.exports = __WEBPACK_EXTERNAL_MODULE__924__;","module.exports = __WEBPACK_EXTERNAL_MODULE__604__;","module.exports = __WEBPACK_EXTERNAL_MODULE__987__;","module.exports = __WEBPACK_EXTERNAL_MODULE__686__;","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","import { type Vector3 } from 'three';\r\n\r\n/**\r\n * Represents a celestial body with all of its kinematic properties.\r\n * @category Building blocks\r\n */\r\nexport class CelestialBody {\r\n  /**\r\n   * Label of the body.\r\n   */\r\n  readonly label: string;\r\n  /**\r\n   * Mean mass of the body.\r\n   */\r\n  readonly mass: number; // mean mass\r\n  /**\r\n   * Position vector of the body.\r\n   */\r\n  position: Vector3;\r\n  /**\r\n   * Velocity vector of the body.\r\n   */\r\n  velocity: Vector3;\r\n  /**\r\n   * Acceleration vector of the body.\r\n   */\r\n  acceleration: Vector3;\r\n\r\n  /**\r\n   * Create a new CelestialBody with the provided information.\r\n   * @param label label of the body.\r\n   * @param mass mass of the body.\r\n   * @param position position of the body.\r\n   * @param velocity velocity of the body.\r\n   * @param acceleration acceleration of the body.\r\n   */\r\n  constructor(\r\n    label: string,\r\n    mass: number,\r\n    position: Vector3,\r\n    velocity: Vector3,\r\n    acceleration: Vector3,\r\n  ) {\r\n    this.label = label;\r\n    this.mass = mass;\r\n    this.position = position;\r\n    this.velocity = velocity;\r\n    this.acceleration = acceleration;\r\n  }\r\n\r\n  /**\r\n   * Deep copy the current CelestialBody with the updated kinematic properties.\r\n   * @param position new position.\r\n   * @param velocity new velocity.\r\n   * @param acceleration new acceleration.\r\n   * @returns a new CelestialBody instance with the updated properties.\r\n   */\r\n  clone(position?: Vector3,\r\n    velocity?: Vector3,\r\n    acceleration?: Vector3): CelestialBody {\r\n    return new CelestialBody(\r\n      this.label,\r\n      this.mass,\r\n      position === undefined ? this.position.clone() : position,\r\n      velocity === undefined ? this.velocity.clone() : velocity,\r\n      acceleration === undefined ? this.acceleration.clone() : acceleration,\r\n    );\r\n  }\r\n}\r\n","import { Vector3 } from 'three';\r\nimport { type CelestialBody } from '../CelestialBody';\r\nimport { type Force } from '../Force';\r\n\r\n/**\r\n * Represents a [Newtonian Gravitational force](https://en.wikipedia.org/wiki/Newton%27s_law_of_universal_gravitation) object.\r\n * @category Forces\r\n */\r\nexport class Gravity implements Force {\r\n  /**\r\n   * Gravitational constant.\r\n   * @defaultValue 6.674e-11\r\n   */\r\n  readonly G: number;\r\n\r\n  /**\r\n   * Create a new Gravity with the provided gravitational constant.\r\n   * @param G gravitational constant.\r\n   */\r\n  constructor(G: number = 6.674e-11) {\r\n    this.G = G;\r\n  }\r\n\r\n  /**\r\n   * Calculate and return the forces acting on the bodies. arr[i] represents the force acting on the ith body as a result of all other bodies.\r\n   * @param bodies celestial bodies\r\n   * @returns forces\r\n   */\r\n  getForces(bodies: CelestialBody[]): Vector3[] {\r\n    let n = bodies.length;\r\n    let ans: Vector3[] = [];\r\n    for (let i = 0; i < n; i++) {\r\n      ans.push(new Vector3(0, 0, 0));\r\n    }\r\n    for (let i = 0; i < n; i++) {\r\n      for (let j = i + 1; j < n; j++) {\r\n        let currForce = this.calcNewtonian(bodies[i], bodies[j]);\r\n        ans[i].add(currForce);\r\n        ans[j].sub(currForce);\r\n      }\r\n    }\r\n    return ans;\r\n  }\r\n\r\n  /**\r\n   * Calculate the gravitational force acting on body a due to body b.\r\n   * @param a body a.\r\n   * @param b body b.\r\n   * @returns gravitational force of a on b.\r\n   * @hidden\r\n   */\r\n  private calcNewtonian(a: CelestialBody, b: CelestialBody): Vector3 {\r\n    let distSq = a.position.distanceToSquared(b.position);\r\n    let forceVal = (this.G * a.mass * b.mass) / distSq;\r\n    return b.position\r\n      .clone()\r\n      .sub(a.position)\r\n      .normalize()\r\n      .multiplyScalar(forceVal);\r\n  }\r\n}\r\n\r\n/**\r\n * Represents a [Centripetal force](https://en.wikipedia.org/wiki/Centripetal_force) object. To be used to calculate the force required to keep the bodies in circular motion around a given center.\r\n * @category Forces\r\n */\r\nexport class CentripetalForce implements Force {\r\n  /**\r\n   * Center of force.\r\n   */\r\n  readonly center: Vector3;\r\n\r\n  /**\r\n   * Create a new CentripetalForce with the provided center of force.\r\n   * @param center center of force.\r\n   */\r\n  constructor(center: Vector3 = new Vector3(0, 0, 0)) {\r\n    this.center = center;\r\n  }\r\n\r\n  /**\r\n   * Calculate the force required to keep the bodies in circular motion around the center. arr[i] represents the centripetal force required for the ith body.\r\n   * @param bodies celestial bodies.\r\n   * @returns forces.\r\n   */\r\n  getForces(bodies: CelestialBody[]): Vector3[] {\r\n    // TODO: mv^2/r sin or cos thetha?\r\n    return bodies.map((body) => {\r\n      const directionVector = this.center.clone()\r\n        .sub(body.position);\r\n      return directionVector.setLength(\r\n        (body.mass * body.velocity.lengthSq()) / directionVector.length(),\r\n      );\r\n    });\r\n  }\r\n}\r\n\r\n/**\r\n * Represents a combined force object. To be used to additively combine multiple forces acting on a system of bodies.\r\n * @category Forces\r\n */\r\nexport class CombinedForce implements Force {\r\n  forces: Force[];\r\n\r\n  /**\r\n   * Create a new CombinedForce with the provided forces.\r\n   * @param forces array of forces.\r\n   */\r\n  constructor(forces: Force[]) {\r\n    this.forces = forces;\r\n  }\r\n\r\n  /**\r\n   * Get the combined forces acting on the bodies. arr[i] represents the combined force acting on the ith body as a result of all force systems.\r\n   * @param bodies celestial bodies.\r\n   * @returns element-wise combined forces.\r\n   */\r\n  getForces(bodies: CelestialBody[]): Vector3[] {\r\n    const forceVal: Vector3[] = bodies.map(() => new Vector3(0, 0, 0));\r\n    this.forces.forEach((force) => {\r\n      force.getForces(bodies)\r\n        .forEach((val, index) => {\r\n          forceVal[index].add(val);\r\n        });\r\n    });\r\n    return forceVal;\r\n  }\r\n}\r\n","import type { CelestialBody } from './CelestialBody';\r\n\r\n/**\r\n * Represents a Universe's state snapshot.\r\n * @category Building blocks\r\n */\r\nexport class State {\r\n  /**\r\n   * Array of celestial bodies that make up this state of the Universe.\r\n   */\r\n  readonly bodies: CelestialBody[];\r\n\r\n  /**\r\n   * Create a new State with the given celestial bodies.\r\n   * @param bodies array of celestial bodies.\r\n   */\r\n  constructor(bodies: CelestialBody[]) {\r\n    this.bodies = bodies;\r\n  }\r\n\r\n  /**\r\n   * Deep copy this state\r\n   * @returns a new State instance.\r\n   */\r\n  clone(): State {\r\n    return new State(this.bodies.map((body) => body.clone()));\r\n  }\r\n}\r\n","import { Vector3 } from 'three';\r\nimport { type CelestialBody } from '../CelestialBody';\r\nimport { type Force } from '../Force';\r\nimport { type SimulateFunction } from '../SimulateFunction';\r\nimport { State } from '../State';\r\n\r\n// export class VerletSim implements SimulateFunction {\r\n//   forceCalculator: Force;\r\n//   prevDeltaT: number | undefined = undefined;\r\n\r\n//   constructor(forceCalculator: Force) {\r\n//     this.forceCalculator = forceCalculator;\r\n//   }\r\n\r\n//   private fromOneState(\r\n//     currState: State,\r\n//     deltaT: number,\r\n//     forces: Vector3[]\r\n//   ): State {\r\n//     let updatedBodies: CelestialBody[] = currState.bodies.map((body, index) => {\r\n//       const currAccel = forces[index].divideScalar(body.mass);\r\n//       const newPos = currAccel\r\n//         .clone()\r\n//         .multiplyScalar(deltaT / 2)\r\n//         .add(body.velocity)\r\n//         .multiplyScalar(deltaT)\r\n//         .add(body.position);\r\n//       return body.clone(\r\n//         newPos,\r\n//         this.verletVel(body.position, newPos, deltaT),\r\n//         currAccel\r\n//       );\r\n//     });\r\n\r\n//     this.prevDeltaT = deltaT;\r\n//     return new State(updatedBodies);\r\n//   }\r\n\r\n//   simulate(deltaT: number, currState: State, prevState?: State): State {\r\n//     if (deltaT <= 0) {\r\n//       return currState.clone();\r\n//     }\r\n\r\n//     let forces = this.forceCalculator.getForces(currState.bodies);\r\n//     if (forces.length !== currState.bodies.length) {\r\n//       console.error(`forces length !== number of bodies`);\r\n//       return currState.clone();\r\n//     }\r\n\r\n//     if (prevState === undefined) {\r\n//       return this.fromOneState(currState, deltaT, forces);\r\n//     }\r\n\r\n//     let updatedBodies: CelestialBody[] = currState.bodies.map((body, index) => {\r\n//       const currAccel = forces[index].divideScalar(body.mass);\r\n//       const newPos = this.verletPos(\r\n//         prevState.bodies[index].position,\r\n//         body.position,\r\n//         currAccel,\r\n//         deltaT\r\n//       );\r\n//       return body.clone(\r\n//         newPos,\r\n//         this.verletVel(body.position, newPos, deltaT),\r\n//         currAccel\r\n//       );\r\n//     });\r\n\r\n//     this.prevDeltaT = deltaT;\r\n//     return new State(updatedBodies);\r\n//   }\r\n\r\n//   verletPos(\r\n//     oldPos: Vector3,\r\n//     currPos: Vector3,\r\n//     currAccel: Vector3,\r\n//     deltaT: number\r\n//   ): Vector3 {\r\n//     const prevDT = this.prevDeltaT === undefined ? deltaT : this.prevDeltaT;\r\n\r\n//     return currAccel\r\n//       .clone()\r\n//       .multiplyScalar((prevDT + deltaT) / 2)\r\n//       .add(currPos.clone().sub(oldPos).divideScalar(prevDT))\r\n//       .multiplyScalar(deltaT)\r\n//       .add(currPos);\r\n//   }\r\n\r\n//   verletVel(currPos: Vector3, newPos: Vector3, deltaT: number): Vector3 {\r\n//     return newPos.clone().sub(currPos).divideScalar(deltaT);\r\n//   }\r\n// }\r\n\r\n/**\r\n * Represents a simulation function object that uses the [Velocity Verlet integration method](https://en.wikipedia.org/wiki/Verlet_integration#Velocity_Verlet) to simulate the motion of bodies.\r\n * @category SimulateFunctions\r\n */\r\nexport class VelocityVerletSim implements SimulateFunction {\r\n  /**\r\n   * Force object to calculate forces on bodies in the Universe.\r\n   */\r\n  forceCalculator: Force;\r\n\r\n  /**\r\n   * Create a new VelocityVerletSim with the provided force calculator, which is invoked on every simulation step.\r\n   * @param forceCalculator force calculator.\r\n   */\r\n  constructor(forceCalculator: Force) {\r\n    this.forceCalculator = forceCalculator;\r\n  }\r\n\r\n  /**\r\n   * Simulate a step in the Universe by using the previous and/or current state and a time step, using the Velocity Verlet integration method.\r\n   * @param deltaT time step.\r\n   * @param currState current state.\r\n   * @returns new state after the simulation step.\r\n   */\r\n  simulate(deltaT: number, currState: State): State {\r\n    if (deltaT <= 0) {\r\n      return currState.clone();\r\n    }\r\n    // x(t + dt) = x(t) + (v(t) * dt) + (0.5 * a(t) * dt * dt)\r\n    let updatedBodies = currState.bodies.map((v) => {\r\n      let updatedBody = v.clone();\r\n      updatedBody.position = this.calcNewPos(\r\n        updatedBody.position,\r\n        updatedBody.velocity,\r\n        updatedBody.acceleration,\r\n        deltaT,\r\n      );\r\n      return updatedBody;\r\n    });\r\n    // a(t + dt) = accel at x(t + dt)\r\n    let newForces = this.forceCalculator.getForces(updatedBodies);\r\n    return new State(\r\n      updatedBodies.map((b: CelestialBody, i: number) => {\r\n        let newAccel = newForces[i].divideScalar(b.mass);\r\n        // v(t + dt) = v(t) + 0.5 * (a(t) + a(t + dt)) * dt\r\n        b.velocity.add(b.acceleration.add(newAccel)\r\n          .multiplyScalar(deltaT / 2));\r\n        b.acceleration = newAccel;\r\n        return b;\r\n      }),\r\n    );\r\n  }\r\n\r\n  /**\r\n   * Calculate the new position of a body in the following way:\r\n   *\r\n   * newPos = currPos + currVel * deltaT + 0.5 * currAccel * deltaT^2.\r\n   * @param currPos current position.\r\n   * @param currVel current velocity.\r\n   * @param currAccel current acceleration.\r\n   * @param deltaT time step.\r\n   * @returns new position.\r\n   * @hidden\r\n   */\r\n  private calcNewPos(\r\n    currPos: Vector3,\r\n    currVel: Vector3,\r\n    currAccel: Vector3,\r\n    deltaT: number,\r\n  ): Vector3 {\r\n    return currPos\r\n      .clone()\r\n      .add(currVel.clone()\r\n        .multiplyScalar(deltaT))\r\n      .add(currAccel.clone()\r\n        .multiplyScalar(deltaT * deltaT * 0.5));\r\n  }\r\n}\r\n\r\n/**\r\n * Represents a simulation function object that uses the [Euler integration method](https://en.wikipedia.org/wiki/Euler_method) to simulate motions of bodies.\r\n * @category SimulateFunctions\r\n */\r\nexport class ExplicitEulerSim implements SimulateFunction {\r\n  /**\r\n   * Force object to calculate forces on bodies in the Universe.\r\n   */\r\n  force: Force;\r\n\r\n  /**\r\n   * Create a new ExplicitEulerSim with the provided force calculator, which is invoked on every simulation step.\r\n   * @param force force calculator.\r\n   */\r\n  constructor(force: Force) {\r\n    this.force = force;\r\n  }\r\n\r\n  /**\r\n   * Simulate a step in the Universe by using the current state and a time step, using the Euler integration method.\r\n   * @param deltaT time step.\r\n   * @param currState current state.\r\n   * @returns new state after the simulation step.\r\n   */\r\n  simulate(\r\n    deltaT: number,\r\n    currState: State,\r\n  ): State {\r\n    const updatedBodies = currState.bodies.map((b) => b.clone(\r\n      // x(t + dt) = x(t) + v(t) * dt\r\n      this.rateUpdate(b.position, b.velocity, deltaT),\r\n      // v(t + dt) = v(t) + a(t) * dt\r\n      this.rateUpdate(b.velocity, b.acceleration, deltaT),\r\n    ));\r\n    const updatedForces = this.force.getForces(updatedBodies);\r\n    updatedBodies.forEach((b, i) => {\r\n      // a(t + dt) = accel at x(t + dt)\r\n      b.acceleration = updatedForces[i].divideScalar(b.mass);\r\n    });\r\n    return new State(updatedBodies);\r\n  }\r\n\r\n  /**\r\n   * Update a vector quantity using the rate of change and a time step.\r\n   * @param prev previous value.\r\n   * @param rate rate of change.\r\n   * @param deltaT time step.\r\n   * @returns updated value.\r\n   * @hidden\r\n   */\r\n  private rateUpdate(prev: Vector3, rate: Vector3, deltaT: number) {\r\n    return rate.clone()\r\n      .multiplyScalar(deltaT)\r\n      .add(prev);\r\n  }\r\n}\r\n\r\n/**\r\n * Represents a simulation function object that uses the [Semi-Implicit Euler integration method](https://en.wikipedia.org/wiki/Semi-implicit_Euler_method) to simulate the motion of bodies.\r\n * @category SimulateFunctions\r\n */\r\nexport class SemiImplicitEulerSim implements SimulateFunction {\r\n  /**\r\n   * Force object to calculate forces on bodies in the Universe.\r\n   */\r\n  force: Force;\r\n\r\n  /**\r\n   * Create a new SemiImplicitEulerSim with the provided force calculator, which is invoked on every simulation step.\r\n   * @param force force calculator.\r\n   */\r\n  constructor(force: Force) {\r\n    this.force = force;\r\n  }\r\n\r\n  /**\r\n   * Simulate a step in the Universe by using the current state and a time step, using the Semi-Implicit Euler integration method.\r\n   * @param deltaT time step.\r\n   * @param currState current state.\r\n   * @returns new state after the simulation step.\r\n   */\r\n  simulate(\r\n    deltaT: number,\r\n    currState: State,\r\n  ): State {\r\n    const updatedBodies = currState.bodies.map((b) => {\r\n      // v(t + dt) = v(t) + a(t) * dt\r\n      const updatedVel = this.rateUpdate(b.velocity, b.acceleration, deltaT);\r\n      return b.clone(\r\n        // x(t + dt) = x(t) + v(t + dt) * dt\r\n        this.rateUpdate(b.position, updatedVel, deltaT),\r\n        updatedVel,\r\n      );\r\n    });\r\n    const updatedForces = this.force.getForces(updatedBodies);\r\n    updatedBodies.forEach((b, i) => {\r\n      // a(t + dt) = accel at x(t + dt)\r\n      b.acceleration = updatedForces[i].divideScalar(b.mass);\r\n    });\r\n    return new State(updatedBodies);\r\n  }\r\n\r\n  /**\r\n   * Update a vector quantity using the rate of change and a time step.\r\n   * @param prev previous value.\r\n   * @param rate rate of change.\r\n   * @param deltaT time step.\r\n   * @returns updated value.\r\n   * @hidden\r\n   */\r\n  private rateUpdate(prev: Vector3, rate: Vector3, deltaT: number) {\r\n    return rate.clone()\r\n      .multiplyScalar(deltaT)\r\n      .add(prev);\r\n  }\r\n}\r\n\r\n/**\r\n * Container for intermediate kinematic rate values in the Runge-Kutta 4 integration method.\r\n */\r\ntype RungeKuttaSteps = {\r\n  /**\r\n   * Intermediate velocity rate values.\r\n   */\r\n  kv: Vector3[];\r\n  /**\r\n   * Intermediate position rate values.\r\n   */\r\n  kx: Vector3[];\r\n};\r\n\r\n/**\r\n * Represents a simulation function object that uses the [Runge-Kutta 4 integration method](https://en.wikipedia.org/wiki/Runge%E2%80%93Kutta_methods) to simulate the motion of bodies.\r\n * @category SimulateFunctions\r\n */\r\nexport class RungeKutta4Sim implements SimulateFunction {\r\n  /**\r\n   * Force object to calculate forces on bodies in the Universe.\r\n   */\r\n  force: Force;\r\n  /**\r\n   * Weights for weighted average.\r\n   */\r\n  weights: number[];\r\n\r\n  /**\r\n   * Create a new RungeKutta4Sim with the provided weights for average force calculator, which is invoked on every simulation step.\r\n   * @param force force calculator.\r\n   * @param weights weights for weighted average.\r\n   */\r\n  constructor(force: Force, weights: number[]) {\r\n    this.force = force;\r\n    if (weights.length !== 4) {\r\n      throw new Error('Weights for RK4 must be of length 4');\r\n    }\r\n    this.weights = weights;\r\n  }\r\n\r\n  /**\r\n   * Simulate a step in the Universe by using the current state and a time step, using the Runge-Kutta 4 integration method.\r\n   * @param deltaT time step.\r\n   * @param currState current state.\r\n   * @returns new state after the simulation step.\r\n   */\r\n  simulate(\r\n    deltaT: number,\r\n    currState: State,\r\n  ): State {\r\n    let bodiesK: RungeKuttaSteps[] = currState.bodies.map((v) => ({\r\n      // k1v = a(x(t))\r\n      kv: [v.acceleration.clone()],\r\n      // k1r = v(t)\r\n      kx: [v.velocity.clone()],\r\n    }));\r\n    // k2v = a(x(t) + k1x * dt / 2)\r\n    const k2v = this.getInterKV(currState.bodies, bodiesK, 0, deltaT / 2);\r\n    // k2x = v(t) + k1v * dt / 2\r\n    const k2x = this.getInterKX(currState.bodies, bodiesK, 0, deltaT / 2);\r\n    bodiesK.forEach((v, i) => {\r\n      v.kv.push(k2v[i]);\r\n      v.kx.push(k2x[i]);\r\n    });\r\n    // k3v = a(x(t) + k2x * dt / 2)\r\n    const k3v = this.getInterKV(currState.bodies, bodiesK, 1, deltaT / 2);\r\n    // k3x = v(t) + k2v * dt / 2\r\n    const k3x = this.getInterKX(currState.bodies, bodiesK, 1, deltaT / 2);\r\n    bodiesK.forEach((v, i) => {\r\n      v.kv.push(k3v[i]);\r\n      v.kx.push(k3x[i]);\r\n    });\r\n    // k3v = a(x(t) + k3x * dt)\r\n    const k4v = this.getInterKV(currState.bodies, bodiesK, 2, deltaT);\r\n    // k4x = v(t) + k3v * dt\r\n    const k4x = this.getInterKX(currState.bodies, bodiesK, 2, deltaT);\r\n    bodiesK.forEach((v, i) => {\r\n      v.kv.push(k4v[i]);\r\n      v.kx.push(k4x[i]);\r\n    });\r\n    const updatedBodies = currState.bodies.map((b, i) => {\r\n      const weightedXRate = new Vector3();\r\n      const weightedVRate = new Vector3();\r\n      bodiesK[i].kx.forEach((v, j) => {\r\n        weightedXRate.add(v.multiplyScalar(this.weights[j]));\r\n      });\r\n      bodiesK[i].kv.forEach((v, j) => {\r\n        weightedVRate.add(v.multiplyScalar(this.weights[j]));\r\n      });\r\n      return b.clone(\r\n        weightedXRate.multiplyScalar(deltaT / 6)\r\n          .add(b.position),\r\n        weightedVRate.multiplyScalar(deltaT / 6)\r\n          .add(b.velocity),\r\n      );\r\n    });\r\n    const updatedForces = this.force.getForces(updatedBodies);\r\n    updatedBodies.forEach((v, i) => {\r\n      v.acceleration = updatedForces[i].divideScalar(v.mass);\r\n    });\r\n    return new State(updatedBodies);\r\n  }\r\n\r\n  /**\r\n   * Get intermediate velocity rate values.\r\n   * @param bodies current bodies.\r\n   * @param bodiesK intermediate kinematic rate values for all bodies.\r\n   * @param krInd index of intermediate kinematic rate value to base the new intermediate value on.\r\n   * @param deltaT time step.\r\n   * @returns next intermediate velocity rate values for all bodies.\r\n   * @hidden\r\n   */\r\n  private getInterKV(\r\n    bodies: CelestialBody[],\r\n    bodiesK: RungeKuttaSteps[],\r\n    krInd: number,\r\n    deltaT: number,\r\n  ): Vector3[] {\r\n    // newX = x(t) + k(kInd)x * delta\r\n    let newBodies = bodies.map((v, i) => {\r\n      let newBody = v.clone();\r\n      newBody.position.add(bodiesK[i].kx[krInd].clone()\r\n        .multiplyScalar(deltaT));\r\n      return newBody;\r\n    });\r\n    // accel at newX\r\n    return this.force.getForces(newBodies)\r\n      .map((v, i) => v.divideScalar(bodies[i].mass));\r\n  }\r\n\r\n  /**\r\n   * Get intermediate position rate values.\r\n   * @param bodies current bodies.\r\n   * @param bodiesK intermediate kinematic rate values for all bodies.\r\n   * @param kInd index of intermediate kinematic rate value to base the new intermediate value on.\r\n   * @param deltaT time step.\r\n   * @returns next intermediate position rate values for all bodies.\r\n   * @hidden\r\n   */\r\n  private getInterKX(\r\n    bodies: CelestialBody[],\r\n    bodiesK: RungeKuttaSteps[],\r\n    kInd: number,\r\n    deltaT: number,\r\n  ): Vector3[] {\r\n    // v(t) + k(kInd)v * deltaT\r\n    return bodies.map((v, i) => v.velocity.clone()\r\n      .add(bodiesK[i].kv[kInd].clone()\r\n        .multiplyScalar(deltaT)));\r\n  }\r\n}\r\n","import type { Force } from './Force';\r\nimport type { State } from './State';\r\n\r\n/**\r\n * Represents a function object used for simulating the Universe. Should encapsulate the numerical integration method and other necessary simulation logic. Can use an external force calculation function object - see {@link Force}.\r\n * @category Interfaces\r\n * @category SimulateFunctions\r\n */\r\nexport interface SimulateFunction {\r\n  /**\r\n   * Simulate a step in the Universe by using the previous and/or current state and a time step.\r\n   * @param deltaT time step.\r\n   * @param currState current state of the Universe.\r\n   * @param prevState previous state of the Universe.\r\n   * @returns the next state of the Universe.\r\n   */\r\n  simulate(deltaT: number, currState: State, prevState: State): State\r\n}\r\n\r\n/**\r\n * Function object that uses the user-defined lambda function to simulate the Universe.\r\n * @category SimulateFunctions\r\n */\r\nexport class LambdaSim implements SimulateFunction {\r\n  readonly fn: (deltaT: number, currState: State, prevState: State) => State;\r\n\r\n  /**\r\n   * Create a new LambdaSim with the provided lambda function.\r\n   * - The lambda function should take in the time step, the current state of the Universe, and the previous state of the Universe, and return the next state of the Universe.\r\n   * - The lambda function should call or calculate the forces action on the bodies by itself.\r\n   * @param fn lambda function.\r\n   */\r\n  constructor(fn: (deltaT: number, currState: State, prevState: State) => State) {\r\n    this.fn = fn;\r\n  }\r\n\r\n  /**\r\n   * Simulate the Universe using the lambda function.\r\n   * @param deltaT time step.\r\n   * @param currState current state of the Universe.\r\n   * @param prevState previous state of the Universe.\r\n   * @returns the next state of the Universe.\r\n   */\r\n  simulate(deltaT: number, currState: State, prevState: State): State {\r\n    return this.fn(deltaT, currState, prevState);\r\n  }\r\n}\r\n","/**\n * lil-gui\n * https://lil-gui.georgealways.com\n * @version 0.19.2\n * @author George Michael Brower\n * @license MIT\n */\n\n/**\n * Base class for all controllers.\n */\nclass Controller {\n\n\tconstructor( parent, object, property, className, elementType = 'div' ) {\n\n\t\t/**\n\t\t * The GUI that contains this controller.\n\t\t * @type {GUI}\n\t\t */\n\t\tthis.parent = parent;\n\n\t\t/**\n\t\t * The object this controller will modify.\n\t\t * @type {object}\n\t\t */\n\t\tthis.object = object;\n\n\t\t/**\n\t\t * The name of the property to control.\n\t\t * @type {string}\n\t\t */\n\t\tthis.property = property;\n\n\t\t/**\n\t\t * Used to determine if the controller is disabled.\n\t\t * Use `controller.disable( true|false )` to modify this value.\n\t\t * @type {boolean}\n\t\t */\n\t\tthis._disabled = false;\n\n\t\t/**\n\t\t * Used to determine if the Controller is hidden.\n\t\t * Use `controller.show()` or `controller.hide()` to change this.\n\t\t * @type {boolean}\n\t\t */\n\t\tthis._hidden = false;\n\n\t\t/**\n\t\t * The value of `object[ property ]` when the controller was created.\n\t\t * @type {any}\n\t\t */\n\t\tthis.initialValue = this.getValue();\n\n\t\t/**\n\t\t * The outermost container DOM element for this controller.\n\t\t * @type {HTMLElement}\n\t\t */\n\t\tthis.domElement = document.createElement( elementType );\n\t\tthis.domElement.classList.add( 'controller' );\n\t\tthis.domElement.classList.add( className );\n\n\t\t/**\n\t\t * The DOM element that contains the controller's name.\n\t\t * @type {HTMLElement}\n\t\t */\n\t\tthis.$name = document.createElement( 'div' );\n\t\tthis.$name.classList.add( 'name' );\n\n\t\tController.nextNameID = Controller.nextNameID || 0;\n\t\tthis.$name.id = `lil-gui-name-${++Controller.nextNameID}`;\n\n\t\t/**\n\t\t * The DOM element that contains the controller's \"widget\" (which differs by controller type).\n\t\t * @type {HTMLElement}\n\t\t */\n\t\tthis.$widget = document.createElement( 'div' );\n\t\tthis.$widget.classList.add( 'widget' );\n\n\t\t/**\n\t\t * The DOM element that receives the disabled attribute when using disable().\n\t\t * @type {HTMLElement}\n\t\t */\n\t\tthis.$disable = this.$widget;\n\n\t\tthis.domElement.appendChild( this.$name );\n\t\tthis.domElement.appendChild( this.$widget );\n\n\t\t// Don't fire global key events while typing in a controller\n\t\tthis.domElement.addEventListener( 'keydown', e => e.stopPropagation() );\n\t\tthis.domElement.addEventListener( 'keyup', e => e.stopPropagation() );\n\n\t\tthis.parent.children.push( this );\n\t\tthis.parent.controllers.push( this );\n\n\t\tthis.parent.$children.appendChild( this.domElement );\n\n\t\tthis._listenCallback = this._listenCallback.bind( this );\n\n\t\tthis.name( property );\n\n\t}\n\n\t/**\n\t * Sets the name of the controller and its label in the GUI.\n\t * @param {string} name\n\t * @returns {this}\n\t */\n\tname( name ) {\n\t\t/**\n\t\t * The controller's name. Use `controller.name( 'Name' )` to modify this value.\n\t\t * @type {string}\n\t\t */\n\t\tthis._name = name;\n\t\tthis.$name.textContent = name;\n\t\treturn this;\n\t}\n\n\t/**\n\t * Pass a function to be called whenever the value is modified by this controller.\n\t * The function receives the new value as its first parameter. The value of `this` will be the\n\t * controller.\n\t *\n\t * For function controllers, the `onChange` callback will be fired on click, after the function\n\t * executes.\n\t * @param {Function} callback\n\t * @returns {this}\n\t * @example\n\t * const controller = gui.add( object, 'property' );\n\t *\n\t * controller.onChange( function( v ) {\n\t * \tconsole.log( 'The value is now ' + v );\n\t * \tconsole.assert( this === controller );\n\t * } );\n\t */\n\tonChange( callback ) {\n\t\t/**\n\t\t * Used to access the function bound to `onChange` events. Don't modify this value directly.\n\t\t * Use the `controller.onChange( callback )` method instead.\n\t\t * @type {Function}\n\t\t */\n\t\tthis._onChange = callback;\n\t\treturn this;\n\t}\n\n\t/**\n\t * Calls the onChange methods of this controller and its parent GUI.\n\t * @protected\n\t */\n\t_callOnChange() {\n\n\t\tthis.parent._callOnChange( this );\n\n\t\tif ( this._onChange !== undefined ) {\n\t\t\tthis._onChange.call( this, this.getValue() );\n\t\t}\n\n\t\tthis._changed = true;\n\n\t}\n\n\t/**\n\t * Pass a function to be called after this controller has been modified and loses focus.\n\t * @param {Function} callback\n\t * @returns {this}\n\t * @example\n\t * const controller = gui.add( object, 'property' );\n\t *\n\t * controller.onFinishChange( function( v ) {\n\t * \tconsole.log( 'Changes complete: ' + v );\n\t * \tconsole.assert( this === controller );\n\t * } );\n\t */\n\tonFinishChange( callback ) {\n\t\t/**\n\t\t * Used to access the function bound to `onFinishChange` events. Don't modify this value\n\t\t * directly. Use the `controller.onFinishChange( callback )` method instead.\n\t\t * @type {Function}\n\t\t */\n\t\tthis._onFinishChange = callback;\n\t\treturn this;\n\t}\n\n\t/**\n\t * Should be called by Controller when its widgets lose focus.\n\t * @protected\n\t */\n\t_callOnFinishChange() {\n\n\t\tif ( this._changed ) {\n\n\t\t\tthis.parent._callOnFinishChange( this );\n\n\t\t\tif ( this._onFinishChange !== undefined ) {\n\t\t\t\tthis._onFinishChange.call( this, this.getValue() );\n\t\t\t}\n\n\t\t}\n\n\t\tthis._changed = false;\n\n\t}\n\n\t/**\n\t * Sets the controller back to its initial value.\n\t * @returns {this}\n\t */\n\treset() {\n\t\tthis.setValue( this.initialValue );\n\t\tthis._callOnFinishChange();\n\t\treturn this;\n\t}\n\n\t/**\n\t * Enables this controller.\n\t * @param {boolean} enabled\n\t * @returns {this}\n\t * @example\n\t * controller.enable();\n\t * controller.enable( false ); // disable\n\t * controller.enable( controller._disabled ); // toggle\n\t */\n\tenable( enabled = true ) {\n\t\treturn this.disable( !enabled );\n\t}\n\n\t/**\n\t * Disables this controller.\n\t * @param {boolean} disabled\n\t * @returns {this}\n\t * @example\n\t * controller.disable();\n\t * controller.disable( false ); // enable\n\t * controller.disable( !controller._disabled ); // toggle\n\t */\n\tdisable( disabled = true ) {\n\n\t\tif ( disabled === this._disabled ) return this;\n\n\t\tthis._disabled = disabled;\n\n\t\tthis.domElement.classList.toggle( 'disabled', disabled );\n\t\tthis.$disable.toggleAttribute( 'disabled', disabled );\n\n\t\treturn this;\n\n\t}\n\n\t/**\n\t * Shows the Controller after it's been hidden.\n\t * @param {boolean} show\n\t * @returns {this}\n\t * @example\n\t * controller.show();\n\t * controller.show( false ); // hide\n\t * controller.show( controller._hidden ); // toggle\n\t */\n\tshow( show = true ) {\n\n\t\tthis._hidden = !show;\n\n\t\tthis.domElement.style.display = this._hidden ? 'none' : '';\n\n\t\treturn this;\n\n\t}\n\n\t/**\n\t * Hides the Controller.\n\t * @returns {this}\n\t */\n\thide() {\n\t\treturn this.show( false );\n\t}\n\n\t/**\n\t * Changes this controller into a dropdown of options.\n\t *\n\t * Calling this method on an option controller will simply update the options. However, if this\n\t * controller was not already an option controller, old references to this controller are\n\t * destroyed, and a new controller is added to the end of the GUI.\n\t * @example\n\t * // safe usage\n\t *\n\t * gui.add( obj, 'prop1' ).options( [ 'a', 'b', 'c' ] );\n\t * gui.add( obj, 'prop2' ).options( { Big: 10, Small: 1 } );\n\t * gui.add( obj, 'prop3' );\n\t *\n\t * // danger\n\t *\n\t * const ctrl1 = gui.add( obj, 'prop1' );\n\t * gui.add( obj, 'prop2' );\n\t *\n\t * // calling options out of order adds a new controller to the end...\n\t * const ctrl2 = ctrl1.options( [ 'a', 'b', 'c' ] );\n\t *\n\t * // ...and ctrl1 now references a controller that doesn't exist\n\t * assert( ctrl2 !== ctrl1 )\n\t * @param {object|Array} options\n\t * @returns {Controller}\n\t */\n\toptions( options ) {\n\t\tconst controller = this.parent.add( this.object, this.property, options );\n\t\tcontroller.name( this._name );\n\t\tthis.destroy();\n\t\treturn controller;\n\t}\n\n\t/**\n\t * Sets the minimum value. Only works on number controllers.\n\t * @param {number} min\n\t * @returns {this}\n\t */\n\tmin( min ) {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Sets the maximum value. Only works on number controllers.\n\t * @param {number} max\n\t * @returns {this}\n\t */\n\tmax( max ) {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Values set by this controller will be rounded to multiples of `step`. Only works on number\n\t * controllers.\n\t * @param {number} step\n\t * @returns {this}\n\t */\n\tstep( step ) {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Rounds the displayed value to a fixed number of decimals, without affecting the actual value\n\t * like `step()`. Only works on number controllers.\n\t * @example\n\t * gui.add( object, 'property' ).listen().decimals( 4 );\n\t * @param {number} decimals\n\t * @returns {this}\n\t */\n\tdecimals( decimals ) {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Calls `updateDisplay()` every animation frame. Pass `false` to stop listening.\n\t * @param {boolean} listen\n\t * @returns {this}\n\t */\n\tlisten( listen = true ) {\n\n\t\t/**\n\t\t * Used to determine if the controller is currently listening. Don't modify this value\n\t\t * directly. Use the `controller.listen( true|false )` method instead.\n\t\t * @type {boolean}\n\t\t */\n\t\tthis._listening = listen;\n\n\t\tif ( this._listenCallbackID !== undefined ) {\n\t\t\tcancelAnimationFrame( this._listenCallbackID );\n\t\t\tthis._listenCallbackID = undefined;\n\t\t}\n\n\t\tif ( this._listening ) {\n\t\t\tthis._listenCallback();\n\t\t}\n\n\t\treturn this;\n\n\t}\n\n\t_listenCallback() {\n\n\t\tthis._listenCallbackID = requestAnimationFrame( this._listenCallback );\n\n\t\t// To prevent framerate loss, make sure the value has changed before updating the display.\n\t\t// Note: save() is used here instead of getValue() only because of ColorController. The !== operator\n\t\t// won't work for color objects or arrays, but ColorController.save() always returns a string.\n\n\t\tconst curValue = this.save();\n\n\t\tif ( curValue !== this._listenPrevValue ) {\n\t\t\tthis.updateDisplay();\n\t\t}\n\n\t\tthis._listenPrevValue = curValue;\n\n\t}\n\n\t/**\n\t * Returns `object[ property ]`.\n\t * @returns {any}\n\t */\n\tgetValue() {\n\t\treturn this.object[ this.property ];\n\t}\n\n\t/**\n\t * Sets the value of `object[ property ]`, invokes any `onChange` handlers and updates the display.\n\t * @param {any} value\n\t * @returns {this}\n\t */\n\tsetValue( value ) {\n\n\t\tif ( this.getValue() !== value ) {\n\n\t\t\tthis.object[ this.property ] = value;\n\t\t\tthis._callOnChange();\n\t\t\tthis.updateDisplay();\n\n\t\t}\n\n\t\treturn this;\n\n\t}\n\n\t/**\n\t * Updates the display to keep it in sync with the current value. Useful for updating your\n\t * controllers when their values have been modified outside of the GUI.\n\t * @returns {this}\n\t */\n\tupdateDisplay() {\n\t\treturn this;\n\t}\n\n\tload( value ) {\n\t\tthis.setValue( value );\n\t\tthis._callOnFinishChange();\n\t\treturn this;\n\t}\n\n\tsave() {\n\t\treturn this.getValue();\n\t}\n\n\t/**\n\t * Destroys this controller and removes it from the parent GUI.\n\t */\n\tdestroy() {\n\t\tthis.listen( false );\n\t\tthis.parent.children.splice( this.parent.children.indexOf( this ), 1 );\n\t\tthis.parent.controllers.splice( this.parent.controllers.indexOf( this ), 1 );\n\t\tthis.parent.$children.removeChild( this.domElement );\n\t}\n\n}\n\nclass BooleanController extends Controller {\n\n\tconstructor( parent, object, property ) {\n\n\t\tsuper( parent, object, property, 'boolean', 'label' );\n\n\t\tthis.$input = document.createElement( 'input' );\n\t\tthis.$input.setAttribute( 'type', 'checkbox' );\n\t\tthis.$input.setAttribute( 'aria-labelledby', this.$name.id );\n\n\t\tthis.$widget.appendChild( this.$input );\n\n\t\tthis.$input.addEventListener( 'change', () => {\n\t\t\tthis.setValue( this.$input.checked );\n\t\t\tthis._callOnFinishChange();\n\t\t} );\n\n\t\tthis.$disable = this.$input;\n\n\t\tthis.updateDisplay();\n\n\t}\n\n\tupdateDisplay() {\n\t\tthis.$input.checked = this.getValue();\n\t\treturn this;\n\t}\n\n}\n\nfunction normalizeColorString( string ) {\n\n\tlet match, result;\n\n\tif ( match = string.match( /(#|0x)?([a-f0-9]{6})/i ) ) {\n\n\t\tresult = match[ 2 ];\n\n\t} else if ( match = string.match( /rgb\\(\\s*(\\d*)\\s*,\\s*(\\d*)\\s*,\\s*(\\d*)\\s*\\)/ ) ) {\n\n\t\tresult = parseInt( match[ 1 ] ).toString( 16 ).padStart( 2, 0 )\n\t\t\t+ parseInt( match[ 2 ] ).toString( 16 ).padStart( 2, 0 )\n\t\t\t+ parseInt( match[ 3 ] ).toString( 16 ).padStart( 2, 0 );\n\n\t} else if ( match = string.match( /^#?([a-f0-9])([a-f0-9])([a-f0-9])$/i ) ) {\n\n\t\tresult = match[ 1 ] + match[ 1 ] + match[ 2 ] + match[ 2 ] + match[ 3 ] + match[ 3 ];\n\n\t}\n\n\tif ( result ) {\n\t\treturn '#' + result;\n\t}\n\n\treturn false;\n\n}\n\nconst STRING = {\n\tisPrimitive: true,\n\tmatch: v => typeof v === 'string',\n\tfromHexString: normalizeColorString,\n\ttoHexString: normalizeColorString\n};\n\nconst INT = {\n\tisPrimitive: true,\n\tmatch: v => typeof v === 'number',\n\tfromHexString: string => parseInt( string.substring( 1 ), 16 ),\n\ttoHexString: value => '#' + value.toString( 16 ).padStart( 6, 0 )\n};\n\nconst ARRAY = {\n\tisPrimitive: false,\n\n\t// The arrow function is here to appease tree shakers like esbuild or webpack.\n\t// See https://esbuild.github.io/api/#tree-shaking\n\tmatch: v => Array.isArray( v ),\n\n\tfromHexString( string, target, rgbScale = 1 ) {\n\n\t\tconst int = INT.fromHexString( string );\n\n\t\ttarget[ 0 ] = ( int >> 16 & 255 ) / 255 * rgbScale;\n\t\ttarget[ 1 ] = ( int >> 8 & 255 ) / 255 * rgbScale;\n\t\ttarget[ 2 ] = ( int & 255 ) / 255 * rgbScale;\n\n\t},\n\ttoHexString( [ r, g, b ], rgbScale = 1 ) {\n\n\t\trgbScale = 255 / rgbScale;\n\n\t\tconst int = ( r * rgbScale ) << 16 ^\n\t\t\t( g * rgbScale ) << 8 ^\n\t\t\t( b * rgbScale ) << 0;\n\n\t\treturn INT.toHexString( int );\n\n\t}\n};\n\nconst OBJECT = {\n\tisPrimitive: false,\n\tmatch: v => Object( v ) === v,\n\tfromHexString( string, target, rgbScale = 1 ) {\n\n\t\tconst int = INT.fromHexString( string );\n\n\t\ttarget.r = ( int >> 16 & 255 ) / 255 * rgbScale;\n\t\ttarget.g = ( int >> 8 & 255 ) / 255 * rgbScale;\n\t\ttarget.b = ( int & 255 ) / 255 * rgbScale;\n\n\t},\n\ttoHexString( { r, g, b }, rgbScale = 1 ) {\n\n\t\trgbScale = 255 / rgbScale;\n\n\t\tconst int = ( r * rgbScale ) << 16 ^\n\t\t\t( g * rgbScale ) << 8 ^\n\t\t\t( b * rgbScale ) << 0;\n\n\t\treturn INT.toHexString( int );\n\n\t}\n};\n\nconst FORMATS = [ STRING, INT, ARRAY, OBJECT ];\n\nfunction getColorFormat( value ) {\n\treturn FORMATS.find( format => format.match( value ) );\n}\n\nclass ColorController extends Controller {\n\n\tconstructor( parent, object, property, rgbScale ) {\n\n\t\tsuper( parent, object, property, 'color' );\n\n\t\tthis.$input = document.createElement( 'input' );\n\t\tthis.$input.setAttribute( 'type', 'color' );\n\t\tthis.$input.setAttribute( 'tabindex', -1 );\n\t\tthis.$input.setAttribute( 'aria-labelledby', this.$name.id );\n\n\t\tthis.$text = document.createElement( 'input' );\n\t\tthis.$text.setAttribute( 'type', 'text' );\n\t\tthis.$text.setAttribute( 'spellcheck', 'false' );\n\t\tthis.$text.setAttribute( 'aria-labelledby', this.$name.id );\n\n\t\tthis.$display = document.createElement( 'div' );\n\t\tthis.$display.classList.add( 'display' );\n\n\t\tthis.$display.appendChild( this.$input );\n\t\tthis.$widget.appendChild( this.$display );\n\t\tthis.$widget.appendChild( this.$text );\n\n\t\tthis._format = getColorFormat( this.initialValue );\n\t\tthis._rgbScale = rgbScale;\n\n\t\tthis._initialValueHexString = this.save();\n\t\tthis._textFocused = false;\n\n\t\tthis.$input.addEventListener( 'input', () => {\n\t\t\tthis._setValueFromHexString( this.$input.value );\n\t\t} );\n\n\t\tthis.$input.addEventListener( 'blur', () => {\n\t\t\tthis._callOnFinishChange();\n\t\t} );\n\n\t\tthis.$text.addEventListener( 'input', () => {\n\t\t\tconst tryParse = normalizeColorString( this.$text.value );\n\t\t\tif ( tryParse ) {\n\t\t\t\tthis._setValueFromHexString( tryParse );\n\t\t\t}\n\t\t} );\n\n\t\tthis.$text.addEventListener( 'focus', () => {\n\t\t\tthis._textFocused = true;\n\t\t\tthis.$text.select();\n\t\t} );\n\n\t\tthis.$text.addEventListener( 'blur', () => {\n\t\t\tthis._textFocused = false;\n\t\t\tthis.updateDisplay();\n\t\t\tthis._callOnFinishChange();\n\t\t} );\n\n\t\tthis.$disable = this.$text;\n\n\t\tthis.updateDisplay();\n\n\t}\n\n\treset() {\n\t\tthis._setValueFromHexString( this._initialValueHexString );\n\t\treturn this;\n\t}\n\n\t_setValueFromHexString( value ) {\n\n\t\tif ( this._format.isPrimitive ) {\n\n\t\t\tconst newValue = this._format.fromHexString( value );\n\t\t\tthis.setValue( newValue );\n\n\t\t} else {\n\n\t\t\tthis._format.fromHexString( value, this.getValue(), this._rgbScale );\n\t\t\tthis._callOnChange();\n\t\t\tthis.updateDisplay();\n\n\t\t}\n\n\t}\n\n\tsave() {\n\t\treturn this._format.toHexString( this.getValue(), this._rgbScale );\n\t}\n\n\tload( value ) {\n\t\tthis._setValueFromHexString( value );\n\t\tthis._callOnFinishChange();\n\t\treturn this;\n\t}\n\n\tupdateDisplay() {\n\t\tthis.$input.value = this._format.toHexString( this.getValue(), this._rgbScale );\n\t\tif ( !this._textFocused ) {\n\t\t\tthis.$text.value = this.$input.value.substring( 1 );\n\t\t}\n\t\tthis.$display.style.backgroundColor = this.$input.value;\n\t\treturn this;\n\t}\n\n}\n\nclass FunctionController extends Controller {\n\n\tconstructor( parent, object, property ) {\n\n\t\tsuper( parent, object, property, 'function' );\n\n\t\t// Buttons are the only case where widget contains name\n\t\tthis.$button = document.createElement( 'button' );\n\t\tthis.$button.appendChild( this.$name );\n\t\tthis.$widget.appendChild( this.$button );\n\n\t\tthis.$button.addEventListener( 'click', e => {\n\t\t\te.preventDefault();\n\t\t\tthis.getValue().call( this.object );\n\t\t\tthis._callOnChange();\n\t\t} );\n\n\t\t// enables :active pseudo class on mobile\n\t\tthis.$button.addEventListener( 'touchstart', () => {}, { passive: true } );\n\n\t\tthis.$disable = this.$button;\n\n\t}\n\n}\n\nclass NumberController extends Controller {\n\n\tconstructor( parent, object, property, min, max, step ) {\n\n\t\tsuper( parent, object, property, 'number' );\n\n\t\tthis._initInput();\n\n\t\tthis.min( min );\n\t\tthis.max( max );\n\n\t\tconst stepExplicit = step !== undefined;\n\t\tthis.step( stepExplicit ? step : this._getImplicitStep(), stepExplicit );\n\n\t\tthis.updateDisplay();\n\n\t}\n\n\tdecimals( decimals ) {\n\t\tthis._decimals = decimals;\n\t\tthis.updateDisplay();\n\t\treturn this;\n\t}\n\n\tmin( min ) {\n\t\tthis._min = min;\n\t\tthis._onUpdateMinMax();\n\t\treturn this;\n\t}\n\n\tmax( max ) {\n\t\tthis._max = max;\n\t\tthis._onUpdateMinMax();\n\t\treturn this;\n\t}\n\n\tstep( step, explicit = true ) {\n\t\tthis._step = step;\n\t\tthis._stepExplicit = explicit;\n\t\treturn this;\n\t}\n\n\tupdateDisplay() {\n\n\t\tconst value = this.getValue();\n\n\t\tif ( this._hasSlider ) {\n\n\t\t\tlet percent = ( value - this._min ) / ( this._max - this._min );\n\t\t\tpercent = Math.max( 0, Math.min( percent, 1 ) );\n\n\t\t\tthis.$fill.style.width = percent * 100 + '%';\n\n\t\t}\n\n\t\tif ( !this._inputFocused ) {\n\t\t\tthis.$input.value = this._decimals === undefined ? value : value.toFixed( this._decimals );\n\t\t}\n\n\t\treturn this;\n\n\t}\n\n\t_initInput() {\n\n\t\tthis.$input = document.createElement( 'input' );\n\t\tthis.$input.setAttribute( 'type', 'text' );\n\t\tthis.$input.setAttribute( 'aria-labelledby', this.$name.id );\n\n\t\t// On touch devices only, use input[type=number] to force a numeric keyboard.\n\t\t// Ideally we could use one input type everywhere, but [type=number] has quirks\n\t\t// on desktop, and [inputmode=decimal] has quirks on iOS.\n\t\t// See https://github.com/georgealways/lil-gui/pull/16\n\n\t\tconst isTouch = window.matchMedia( '(pointer: coarse)' ).matches;\n\n\t\tif ( isTouch ) {\n\t\t\tthis.$input.setAttribute( 'type', 'number' );\n\t\t\tthis.$input.setAttribute( 'step', 'any' );\n\t\t}\n\n\t\tthis.$widget.appendChild( this.$input );\n\n\t\tthis.$disable = this.$input;\n\n\t\tconst onInput = () => {\n\n\t\t\tlet value = parseFloat( this.$input.value );\n\n\t\t\tif ( isNaN( value ) ) return;\n\n\t\t\tif ( this._stepExplicit ) {\n\t\t\t\tvalue = this._snap( value );\n\t\t\t}\n\n\t\t\tthis.setValue( this._clamp( value ) );\n\n\t\t};\n\n\t\t// Keys & mouse wheel\n\t\t// ---------------------------------------------------------------------\n\n\t\tconst increment = delta => {\n\n\t\t\tconst value = parseFloat( this.$input.value );\n\n\t\t\tif ( isNaN( value ) ) return;\n\n\t\t\tthis._snapClampSetValue( value + delta );\n\n\t\t\t// Force the input to updateDisplay when it's focused\n\t\t\tthis.$input.value = this.getValue();\n\n\t\t};\n\n\t\tconst onKeyDown = e => {\n\t\t\t// Using `e.key` instead of `e.code` also catches NumpadEnter\n\t\t\tif ( e.key === 'Enter' ) {\n\t\t\t\tthis.$input.blur();\n\t\t\t}\n\t\t\tif ( e.code === 'ArrowUp' ) {\n\t\t\t\te.preventDefault();\n\t\t\t\tincrement( this._step * this._arrowKeyMultiplier( e ) );\n\t\t\t}\n\t\t\tif ( e.code === 'ArrowDown' ) {\n\t\t\t\te.preventDefault();\n\t\t\t\tincrement( this._step * this._arrowKeyMultiplier( e ) * -1 );\n\t\t\t}\n\t\t};\n\n\t\tconst onWheel = e => {\n\t\t\tif ( this._inputFocused ) {\n\t\t\t\te.preventDefault();\n\t\t\t\tincrement( this._step * this._normalizeMouseWheel( e ) );\n\t\t\t}\n\t\t};\n\n\t\t// Vertical drag\n\t\t// ---------------------------------------------------------------------\n\n\t\tlet testingForVerticalDrag = false,\n\t\t\tinitClientX,\n\t\t\tinitClientY,\n\t\t\tprevClientY,\n\t\t\tinitValue,\n\t\t\tdragDelta;\n\n\t\t// Once the mouse is dragged more than DRAG_THRESH px on any axis, we decide\n\t\t// on the user's intent: horizontal means highlight, vertical means drag.\n\t\tconst DRAG_THRESH = 5;\n\n\t\tconst onMouseDown = e => {\n\n\t\t\tinitClientX = e.clientX;\n\t\t\tinitClientY = prevClientY = e.clientY;\n\t\t\ttestingForVerticalDrag = true;\n\n\t\t\tinitValue = this.getValue();\n\t\t\tdragDelta = 0;\n\n\t\t\twindow.addEventListener( 'mousemove', onMouseMove );\n\t\t\twindow.addEventListener( 'mouseup', onMouseUp );\n\n\t\t};\n\n\t\tconst onMouseMove = e => {\n\n\t\t\tif ( testingForVerticalDrag ) {\n\n\t\t\t\tconst dx = e.clientX - initClientX;\n\t\t\t\tconst dy = e.clientY - initClientY;\n\n\t\t\t\tif ( Math.abs( dy ) > DRAG_THRESH ) {\n\n\t\t\t\t\te.preventDefault();\n\t\t\t\t\tthis.$input.blur();\n\t\t\t\t\ttestingForVerticalDrag = false;\n\t\t\t\t\tthis._setDraggingStyle( true, 'vertical' );\n\n\t\t\t\t} else if ( Math.abs( dx ) > DRAG_THRESH ) {\n\n\t\t\t\t\tonMouseUp();\n\n\t\t\t\t}\n\n\t\t\t}\n\n\t\t\t// This isn't an else so that the first move counts towards dragDelta\n\t\t\tif ( !testingForVerticalDrag ) {\n\n\t\t\t\tconst dy = e.clientY - prevClientY;\n\n\t\t\t\tdragDelta -= dy * this._step * this._arrowKeyMultiplier( e );\n\n\t\t\t\t// Clamp dragDelta so we don't have 'dead space' after dragging past bounds.\n\t\t\t\t// We're okay with the fact that bounds can be undefined here.\n\t\t\t\tif ( initValue + dragDelta > this._max ) {\n\t\t\t\t\tdragDelta = this._max - initValue;\n\t\t\t\t} else if ( initValue + dragDelta < this._min ) {\n\t\t\t\t\tdragDelta = this._min - initValue;\n\t\t\t\t}\n\n\t\t\t\tthis._snapClampSetValue( initValue + dragDelta );\n\n\t\t\t}\n\n\t\t\tprevClientY = e.clientY;\n\n\t\t};\n\n\t\tconst onMouseUp = () => {\n\t\t\tthis._setDraggingStyle( false, 'vertical' );\n\t\t\tthis._callOnFinishChange();\n\t\t\twindow.removeEventListener( 'mousemove', onMouseMove );\n\t\t\twindow.removeEventListener( 'mouseup', onMouseUp );\n\t\t};\n\n\t\t// Focus state & onFinishChange\n\t\t// ---------------------------------------------------------------------\n\n\t\tconst onFocus = () => {\n\t\t\tthis._inputFocused = true;\n\t\t};\n\n\t\tconst onBlur = () => {\n\t\t\tthis._inputFocused = false;\n\t\t\tthis.updateDisplay();\n\t\t\tthis._callOnFinishChange();\n\t\t};\n\n\t\tthis.$input.addEventListener( 'input', onInput );\n\t\tthis.$input.addEventListener( 'keydown', onKeyDown );\n\t\tthis.$input.addEventListener( 'wheel', onWheel, { passive: false } );\n\t\tthis.$input.addEventListener( 'mousedown', onMouseDown );\n\t\tthis.$input.addEventListener( 'focus', onFocus );\n\t\tthis.$input.addEventListener( 'blur', onBlur );\n\n\t}\n\n\t_initSlider() {\n\n\t\tthis._hasSlider = true;\n\n\t\t// Build DOM\n\t\t// ---------------------------------------------------------------------\n\n\t\tthis.$slider = document.createElement( 'div' );\n\t\tthis.$slider.classList.add( 'slider' );\n\n\t\tthis.$fill = document.createElement( 'div' );\n\t\tthis.$fill.classList.add( 'fill' );\n\n\t\tthis.$slider.appendChild( this.$fill );\n\t\tthis.$widget.insertBefore( this.$slider, this.$input );\n\n\t\tthis.domElement.classList.add( 'hasSlider' );\n\n\t\t// Map clientX to value\n\t\t// ---------------------------------------------------------------------\n\n\t\tconst map = ( v, a, b, c, d ) => {\n\t\t\treturn ( v - a ) / ( b - a ) * ( d - c ) + c;\n\t\t};\n\n\t\tconst setValueFromX = clientX => {\n\t\t\tconst rect = this.$slider.getBoundingClientRect();\n\t\t\tlet value = map( clientX, rect.left, rect.right, this._min, this._max );\n\t\t\tthis._snapClampSetValue( value );\n\t\t};\n\n\t\t// Mouse drag\n\t\t// ---------------------------------------------------------------------\n\n\t\tconst mouseDown = e => {\n\t\t\tthis._setDraggingStyle( true );\n\t\t\tsetValueFromX( e.clientX );\n\t\t\twindow.addEventListener( 'mousemove', mouseMove );\n\t\t\twindow.addEventListener( 'mouseup', mouseUp );\n\t\t};\n\n\t\tconst mouseMove = e => {\n\t\t\tsetValueFromX( e.clientX );\n\t\t};\n\n\t\tconst mouseUp = () => {\n\t\t\tthis._callOnFinishChange();\n\t\t\tthis._setDraggingStyle( false );\n\t\t\twindow.removeEventListener( 'mousemove', mouseMove );\n\t\t\twindow.removeEventListener( 'mouseup', mouseUp );\n\t\t};\n\n\t\t// Touch drag\n\t\t// ---------------------------------------------------------------------\n\n\t\tlet testingForScroll = false, prevClientX, prevClientY;\n\n\t\tconst beginTouchDrag = e => {\n\t\t\te.preventDefault();\n\t\t\tthis._setDraggingStyle( true );\n\t\t\tsetValueFromX( e.touches[ 0 ].clientX );\n\t\t\ttestingForScroll = false;\n\t\t};\n\n\t\tconst onTouchStart = e => {\n\n\t\t\tif ( e.touches.length > 1 ) return;\n\n\t\t\t// If we're in a scrollable container, we should wait for the first\n\t\t\t// touchmove to see if the user is trying to slide or scroll.\n\t\t\tif ( this._hasScrollBar ) {\n\n\t\t\t\tprevClientX = e.touches[ 0 ].clientX;\n\t\t\t\tprevClientY = e.touches[ 0 ].clientY;\n\t\t\t\ttestingForScroll = true;\n\n\t\t\t} else {\n\n\t\t\t\t// Otherwise, we can set the value straight away on touchstart.\n\t\t\t\tbeginTouchDrag( e );\n\n\t\t\t}\n\n\t\t\twindow.addEventListener( 'touchmove', onTouchMove, { passive: false } );\n\t\t\twindow.addEventListener( 'touchend', onTouchEnd );\n\n\t\t};\n\n\t\tconst onTouchMove = e => {\n\n\t\t\tif ( testingForScroll ) {\n\n\t\t\t\tconst dx = e.touches[ 0 ].clientX - prevClientX;\n\t\t\t\tconst dy = e.touches[ 0 ].clientY - prevClientY;\n\n\t\t\t\tif ( Math.abs( dx ) > Math.abs( dy ) ) {\n\n\t\t\t\t\t// We moved horizontally, set the value and stop checking.\n\t\t\t\t\tbeginTouchDrag( e );\n\n\t\t\t\t} else {\n\n\t\t\t\t\t// This was, in fact, an attempt to scroll. Abort.\n\t\t\t\t\twindow.removeEventListener( 'touchmove', onTouchMove );\n\t\t\t\t\twindow.removeEventListener( 'touchend', onTouchEnd );\n\n\t\t\t\t}\n\n\t\t\t} else {\n\n\t\t\t\te.preventDefault();\n\t\t\t\tsetValueFromX( e.touches[ 0 ].clientX );\n\n\t\t\t}\n\n\t\t};\n\n\t\tconst onTouchEnd = () => {\n\t\t\tthis._callOnFinishChange();\n\t\t\tthis._setDraggingStyle( false );\n\t\t\twindow.removeEventListener( 'touchmove', onTouchMove );\n\t\t\twindow.removeEventListener( 'touchend', onTouchEnd );\n\t\t};\n\n\t\t// Mouse wheel\n\t\t// ---------------------------------------------------------------------\n\n\t\t// We have to use a debounced function to call onFinishChange because\n\t\t// there's no way to tell when the user is \"done\" mouse-wheeling.\n\t\tconst callOnFinishChange = this._callOnFinishChange.bind( this );\n\t\tconst WHEEL_DEBOUNCE_TIME = 400;\n\t\tlet wheelFinishChangeTimeout;\n\n\t\tconst onWheel = e => {\n\n\t\t\t// ignore vertical wheels if there's a scrollbar\n\t\t\tconst isVertical = Math.abs( e.deltaX ) < Math.abs( e.deltaY );\n\t\t\tif ( isVertical && this._hasScrollBar ) return;\n\n\t\t\te.preventDefault();\n\n\t\t\t// set value\n\t\t\tconst delta = this._normalizeMouseWheel( e ) * this._step;\n\t\t\tthis._snapClampSetValue( this.getValue() + delta );\n\n\t\t\t// force the input to updateDisplay when it's focused\n\t\t\tthis.$input.value = this.getValue();\n\n\t\t\t// debounce onFinishChange\n\t\t\tclearTimeout( wheelFinishChangeTimeout );\n\t\t\twheelFinishChangeTimeout = setTimeout( callOnFinishChange, WHEEL_DEBOUNCE_TIME );\n\n\t\t};\n\n\t\tthis.$slider.addEventListener( 'mousedown', mouseDown );\n\t\tthis.$slider.addEventListener( 'touchstart', onTouchStart, { passive: false } );\n\t\tthis.$slider.addEventListener( 'wheel', onWheel, { passive: false } );\n\n\t}\n\n\t_setDraggingStyle( active, axis = 'horizontal' ) {\n\t\tif ( this.$slider ) {\n\t\t\tthis.$slider.classList.toggle( 'active', active );\n\t\t}\n\t\tdocument.body.classList.toggle( 'lil-gui-dragging', active );\n\t\tdocument.body.classList.toggle( `lil-gui-${axis}`, active );\n\t}\n\n\t_getImplicitStep() {\n\n\t\tif ( this._hasMin && this._hasMax ) {\n\t\t\treturn ( this._max - this._min ) / 1000;\n\t\t}\n\n\t\treturn 0.1;\n\n\t}\n\n\t_onUpdateMinMax() {\n\n\t\tif ( !this._hasSlider && this._hasMin && this._hasMax ) {\n\n\t\t\t// If this is the first time we're hearing about min and max\n\t\t\t// and we haven't explicitly stated what our step is, let's\n\t\t\t// update that too.\n\t\t\tif ( !this._stepExplicit ) {\n\t\t\t\tthis.step( this._getImplicitStep(), false );\n\t\t\t}\n\n\t\t\tthis._initSlider();\n\t\t\tthis.updateDisplay();\n\n\t\t}\n\n\t}\n\n\t_normalizeMouseWheel( e ) {\n\n\t\tlet { deltaX, deltaY } = e;\n\n\t\t// Safari and Chrome report weird non-integral values for a notched wheel,\n\t\t// but still expose actual lines scrolled via wheelDelta. Notched wheels\n\t\t// should behave the same way as arrow keys.\n\t\tif ( Math.floor( e.deltaY ) !== e.deltaY && e.wheelDelta ) {\n\t\t\tdeltaX = 0;\n\t\t\tdeltaY = -e.wheelDelta / 120;\n\t\t\tdeltaY *= this._stepExplicit ? 1 : 10;\n\t\t}\n\n\t\tconst wheel = deltaX + -deltaY;\n\n\t\treturn wheel;\n\n\t}\n\n\t_arrowKeyMultiplier( e ) {\n\n\t\tlet mult = this._stepExplicit ? 1 : 10;\n\n\t\tif ( e.shiftKey ) {\n\t\t\tmult *= 10;\n\t\t} else if ( e.altKey ) {\n\t\t\tmult /= 10;\n\t\t}\n\n\t\treturn mult;\n\n\t}\n\n\t_snap( value ) {\n\n\t\t// This would be the logical way to do things, but floating point errors.\n\t\t// return Math.round( value / this._step ) * this._step;\n\n\t\t// Using inverse step solves a lot of them, but not all\n\t\t// const inverseStep = 1 / this._step;\n\t\t// return Math.round( value * inverseStep ) / inverseStep;\n\n\t\t// Not happy about this, but haven't seen it break.\n\t\tconst r = Math.round( value / this._step ) * this._step;\n\t\treturn parseFloat( r.toPrecision( 15 ) );\n\n\t}\n\n\t_clamp( value ) {\n\t\t// either condition is false if min or max is undefined\n\t\tif ( value < this._min ) value = this._min;\n\t\tif ( value > this._max ) value = this._max;\n\t\treturn value;\n\t}\n\n\t_snapClampSetValue( value ) {\n\t\tthis.setValue( this._clamp( this._snap( value ) ) );\n\t}\n\n\tget _hasScrollBar() {\n\t\tconst root = this.parent.root.$children;\n\t\treturn root.scrollHeight > root.clientHeight;\n\t}\n\n\tget _hasMin() {\n\t\treturn this._min !== undefined;\n\t}\n\n\tget _hasMax() {\n\t\treturn this._max !== undefined;\n\t}\n\n}\n\nclass OptionController extends Controller {\n\n\tconstructor( parent, object, property, options ) {\n\n\t\tsuper( parent, object, property, 'option' );\n\n\t\tthis.$select = document.createElement( 'select' );\n\t\tthis.$select.setAttribute( 'aria-labelledby', this.$name.id );\n\n\t\tthis.$display = document.createElement( 'div' );\n\t\tthis.$display.classList.add( 'display' );\n\n\t\tthis.$select.addEventListener( 'change', () => {\n\t\t\tthis.setValue( this._values[ this.$select.selectedIndex ] );\n\t\t\tthis._callOnFinishChange();\n\t\t} );\n\n\t\tthis.$select.addEventListener( 'focus', () => {\n\t\t\tthis.$display.classList.add( 'focus' );\n\t\t} );\n\n\t\tthis.$select.addEventListener( 'blur', () => {\n\t\t\tthis.$display.classList.remove( 'focus' );\n\t\t} );\n\n\t\tthis.$widget.appendChild( this.$select );\n\t\tthis.$widget.appendChild( this.$display );\n\n\t\tthis.$disable = this.$select;\n\n\t\tthis.options( options );\n\n\t}\n\n\toptions( options ) {\n\n\t\tthis._values = Array.isArray( options ) ? options : Object.values( options );\n\t\tthis._names = Array.isArray( options ) ? options : Object.keys( options );\n\n\t\tthis.$select.replaceChildren();\n\n\t\tthis._names.forEach( name => {\n\t\t\tconst $option = document.createElement( 'option' );\n\t\t\t$option.textContent = name;\n\t\t\tthis.$select.appendChild( $option );\n\t\t} );\n\n\t\tthis.updateDisplay();\n\n\t\treturn this;\n\n\t}\n\n\tupdateDisplay() {\n\t\tconst value = this.getValue();\n\t\tconst index = this._values.indexOf( value );\n\t\tthis.$select.selectedIndex = index;\n\t\tthis.$display.textContent = index === -1 ? value : this._names[ index ];\n\t\treturn this;\n\t}\n\n}\n\nclass StringController extends Controller {\n\n\tconstructor( parent, object, property ) {\n\n\t\tsuper( parent, object, property, 'string' );\n\n\t\tthis.$input = document.createElement( 'input' );\n\t\tthis.$input.setAttribute( 'type', 'text' );\n\t\tthis.$input.setAttribute( 'spellcheck', 'false' );\n\t\tthis.$input.setAttribute( 'aria-labelledby', this.$name.id );\n\n\t\tthis.$input.addEventListener( 'input', () => {\n\t\t\tthis.setValue( this.$input.value );\n\t\t} );\n\n\t\tthis.$input.addEventListener( 'keydown', e => {\n\t\t\tif ( e.code === 'Enter' ) {\n\t\t\t\tthis.$input.blur();\n\t\t\t}\n\t\t} );\n\n\t\tthis.$input.addEventListener( 'blur', () => {\n\t\t\tthis._callOnFinishChange();\n\t\t} );\n\n\t\tthis.$widget.appendChild( this.$input );\n\n\t\tthis.$disable = this.$input;\n\n\t\tthis.updateDisplay();\n\n\t}\n\n\tupdateDisplay() {\n\t\tthis.$input.value = this.getValue();\n\t\treturn this;\n\t}\n\n}\n\nconst stylesheet = `.lil-gui {\n  font-family: var(--font-family);\n  font-size: var(--font-size);\n  line-height: 1;\n  font-weight: normal;\n  font-style: normal;\n  text-align: left;\n  color: var(--text-color);\n  user-select: none;\n  -webkit-user-select: none;\n  touch-action: manipulation;\n  --background-color: #1f1f1f;\n  --text-color: #ebebeb;\n  --title-background-color: #111111;\n  --title-text-color: #ebebeb;\n  --widget-color: #424242;\n  --hover-color: #4f4f4f;\n  --focus-color: #595959;\n  --number-color: #2cc9ff;\n  --string-color: #a2db3c;\n  --font-size: 11px;\n  --input-font-size: 11px;\n  --font-family: -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Arial, sans-serif;\n  --font-family-mono: Menlo, Monaco, Consolas, \"Droid Sans Mono\", monospace;\n  --padding: 4px;\n  --spacing: 4px;\n  --widget-height: 20px;\n  --title-height: calc(var(--widget-height) + var(--spacing) * 1.25);\n  --name-width: 45%;\n  --slider-knob-width: 2px;\n  --slider-input-width: 27%;\n  --color-input-width: 27%;\n  --slider-input-min-width: 45px;\n  --color-input-min-width: 45px;\n  --folder-indent: 7px;\n  --widget-padding: 0 0 0 3px;\n  --widget-border-radius: 2px;\n  --checkbox-size: calc(0.75 * var(--widget-height));\n  --scrollbar-width: 5px;\n}\n.lil-gui, .lil-gui * {\n  box-sizing: border-box;\n  margin: 0;\n  padding: 0;\n}\n.lil-gui.root {\n  width: var(--width, 245px);\n  display: flex;\n  flex-direction: column;\n  background: var(--background-color);\n}\n.lil-gui.root > .title {\n  background: var(--title-background-color);\n  color: var(--title-text-color);\n}\n.lil-gui.root > .children {\n  overflow-x: hidden;\n  overflow-y: auto;\n}\n.lil-gui.root > .children::-webkit-scrollbar {\n  width: var(--scrollbar-width);\n  height: var(--scrollbar-width);\n  background: var(--background-color);\n}\n.lil-gui.root > .children::-webkit-scrollbar-thumb {\n  border-radius: var(--scrollbar-width);\n  background: var(--focus-color);\n}\n@media (pointer: coarse) {\n  .lil-gui.allow-touch-styles, .lil-gui.allow-touch-styles .lil-gui {\n    --widget-height: 28px;\n    --padding: 6px;\n    --spacing: 6px;\n    --font-size: 13px;\n    --input-font-size: 16px;\n    --folder-indent: 10px;\n    --scrollbar-width: 7px;\n    --slider-input-min-width: 50px;\n    --color-input-min-width: 65px;\n  }\n}\n.lil-gui.force-touch-styles, .lil-gui.force-touch-styles .lil-gui {\n  --widget-height: 28px;\n  --padding: 6px;\n  --spacing: 6px;\n  --font-size: 13px;\n  --input-font-size: 16px;\n  --folder-indent: 10px;\n  --scrollbar-width: 7px;\n  --slider-input-min-width: 50px;\n  --color-input-min-width: 65px;\n}\n.lil-gui.autoPlace {\n  max-height: 100%;\n  position: fixed;\n  top: 0;\n  right: 15px;\n  z-index: 1001;\n}\n\n.lil-gui .controller {\n  display: flex;\n  align-items: center;\n  padding: 0 var(--padding);\n  margin: var(--spacing) 0;\n}\n.lil-gui .controller.disabled {\n  opacity: 0.5;\n}\n.lil-gui .controller.disabled, .lil-gui .controller.disabled * {\n  pointer-events: none !important;\n}\n.lil-gui .controller > .name {\n  min-width: var(--name-width);\n  flex-shrink: 0;\n  white-space: pre;\n  padding-right: var(--spacing);\n  line-height: var(--widget-height);\n}\n.lil-gui .controller .widget {\n  position: relative;\n  display: flex;\n  align-items: center;\n  width: 100%;\n  min-height: var(--widget-height);\n}\n.lil-gui .controller.string input {\n  color: var(--string-color);\n}\n.lil-gui .controller.boolean {\n  cursor: pointer;\n}\n.lil-gui .controller.color .display {\n  width: 100%;\n  height: var(--widget-height);\n  border-radius: var(--widget-border-radius);\n  position: relative;\n}\n@media (hover: hover) {\n  .lil-gui .controller.color .display:hover:before {\n    content: \" \";\n    display: block;\n    position: absolute;\n    border-radius: var(--widget-border-radius);\n    border: 1px solid #fff9;\n    top: 0;\n    right: 0;\n    bottom: 0;\n    left: 0;\n  }\n}\n.lil-gui .controller.color input[type=color] {\n  opacity: 0;\n  width: 100%;\n  height: 100%;\n  cursor: pointer;\n}\n.lil-gui .controller.color input[type=text] {\n  margin-left: var(--spacing);\n  font-family: var(--font-family-mono);\n  min-width: var(--color-input-min-width);\n  width: var(--color-input-width);\n  flex-shrink: 0;\n}\n.lil-gui .controller.option select {\n  opacity: 0;\n  position: absolute;\n  width: 100%;\n  max-width: 100%;\n}\n.lil-gui .controller.option .display {\n  position: relative;\n  pointer-events: none;\n  border-radius: var(--widget-border-radius);\n  height: var(--widget-height);\n  line-height: var(--widget-height);\n  max-width: 100%;\n  overflow: hidden;\n  word-break: break-all;\n  padding-left: 0.55em;\n  padding-right: 1.75em;\n  background: var(--widget-color);\n}\n@media (hover: hover) {\n  .lil-gui .controller.option .display.focus {\n    background: var(--focus-color);\n  }\n}\n.lil-gui .controller.option .display.active {\n  background: var(--focus-color);\n}\n.lil-gui .controller.option .display:after {\n  font-family: \"lil-gui\";\n  content: \"↕\";\n  position: absolute;\n  top: 0;\n  right: 0;\n  bottom: 0;\n  padding-right: 0.375em;\n}\n.lil-gui .controller.option .widget,\n.lil-gui .controller.option select {\n  cursor: pointer;\n}\n@media (hover: hover) {\n  .lil-gui .controller.option .widget:hover .display {\n    background: var(--hover-color);\n  }\n}\n.lil-gui .controller.number input {\n  color: var(--number-color);\n}\n.lil-gui .controller.number.hasSlider input {\n  margin-left: var(--spacing);\n  width: var(--slider-input-width);\n  min-width: var(--slider-input-min-width);\n  flex-shrink: 0;\n}\n.lil-gui .controller.number .slider {\n  width: 100%;\n  height: var(--widget-height);\n  background: var(--widget-color);\n  border-radius: var(--widget-border-radius);\n  padding-right: var(--slider-knob-width);\n  overflow: hidden;\n  cursor: ew-resize;\n  touch-action: pan-y;\n}\n@media (hover: hover) {\n  .lil-gui .controller.number .slider:hover {\n    background: var(--hover-color);\n  }\n}\n.lil-gui .controller.number .slider.active {\n  background: var(--focus-color);\n}\n.lil-gui .controller.number .slider.active .fill {\n  opacity: 0.95;\n}\n.lil-gui .controller.number .fill {\n  height: 100%;\n  border-right: var(--slider-knob-width) solid var(--number-color);\n  box-sizing: content-box;\n}\n\n.lil-gui-dragging .lil-gui {\n  --hover-color: var(--widget-color);\n}\n.lil-gui-dragging * {\n  cursor: ew-resize !important;\n}\n\n.lil-gui-dragging.lil-gui-vertical * {\n  cursor: ns-resize !important;\n}\n\n.lil-gui .title {\n  height: var(--title-height);\n  line-height: calc(var(--title-height) - 4px);\n  font-weight: 600;\n  padding: 0 var(--padding);\n  -webkit-tap-highlight-color: transparent;\n  cursor: pointer;\n  outline: none;\n  text-decoration-skip: objects;\n}\n.lil-gui .title:before {\n  font-family: \"lil-gui\";\n  content: \"▾\";\n  padding-right: 2px;\n  display: inline-block;\n}\n.lil-gui .title:active {\n  background: var(--title-background-color);\n  opacity: 0.75;\n}\n@media (hover: hover) {\n  body:not(.lil-gui-dragging) .lil-gui .title:hover {\n    background: var(--title-background-color);\n    opacity: 0.85;\n  }\n  .lil-gui .title:focus {\n    text-decoration: underline var(--focus-color);\n  }\n}\n.lil-gui.root > .title:focus {\n  text-decoration: none !important;\n}\n.lil-gui.closed > .title:before {\n  content: \"▸\";\n}\n.lil-gui.closed > .children {\n  transform: translateY(-7px);\n  opacity: 0;\n}\n.lil-gui.closed:not(.transition) > .children {\n  display: none;\n}\n.lil-gui.transition > .children {\n  transition-duration: 300ms;\n  transition-property: height, opacity, transform;\n  transition-timing-function: cubic-bezier(0.2, 0.6, 0.35, 1);\n  overflow: hidden;\n  pointer-events: none;\n}\n.lil-gui .children:empty:before {\n  content: \"Empty\";\n  padding: 0 var(--padding);\n  margin: var(--spacing) 0;\n  display: block;\n  height: var(--widget-height);\n  font-style: italic;\n  line-height: var(--widget-height);\n  opacity: 0.5;\n}\n.lil-gui.root > .children > .lil-gui > .title {\n  border: 0 solid var(--widget-color);\n  border-width: 1px 0;\n  transition: border-color 300ms;\n}\n.lil-gui.root > .children > .lil-gui.closed > .title {\n  border-bottom-color: transparent;\n}\n.lil-gui + .controller {\n  border-top: 1px solid var(--widget-color);\n  margin-top: 0;\n  padding-top: var(--spacing);\n}\n.lil-gui .lil-gui .lil-gui > .title {\n  border: none;\n}\n.lil-gui .lil-gui .lil-gui > .children {\n  border: none;\n  margin-left: var(--folder-indent);\n  border-left: 2px solid var(--widget-color);\n}\n.lil-gui .lil-gui .controller {\n  border: none;\n}\n\n.lil-gui label, .lil-gui input, .lil-gui button {\n  -webkit-tap-highlight-color: transparent;\n}\n.lil-gui input {\n  border: 0;\n  outline: none;\n  font-family: var(--font-family);\n  font-size: var(--input-font-size);\n  border-radius: var(--widget-border-radius);\n  height: var(--widget-height);\n  background: var(--widget-color);\n  color: var(--text-color);\n  width: 100%;\n}\n@media (hover: hover) {\n  .lil-gui input:hover {\n    background: var(--hover-color);\n  }\n  .lil-gui input:active {\n    background: var(--focus-color);\n  }\n}\n.lil-gui input:disabled {\n  opacity: 1;\n}\n.lil-gui input[type=text],\n.lil-gui input[type=number] {\n  padding: var(--widget-padding);\n  -moz-appearance: textfield;\n}\n.lil-gui input[type=text]:focus,\n.lil-gui input[type=number]:focus {\n  background: var(--focus-color);\n}\n.lil-gui input[type=checkbox] {\n  appearance: none;\n  width: var(--checkbox-size);\n  height: var(--checkbox-size);\n  border-radius: var(--widget-border-radius);\n  text-align: center;\n  cursor: pointer;\n}\n.lil-gui input[type=checkbox]:checked:before {\n  font-family: \"lil-gui\";\n  content: \"✓\";\n  font-size: var(--checkbox-size);\n  line-height: var(--checkbox-size);\n}\n@media (hover: hover) {\n  .lil-gui input[type=checkbox]:focus {\n    box-shadow: inset 0 0 0 1px var(--focus-color);\n  }\n}\n.lil-gui button {\n  outline: none;\n  cursor: pointer;\n  font-family: var(--font-family);\n  font-size: var(--font-size);\n  color: var(--text-color);\n  width: 100%;\n  height: var(--widget-height);\n  text-transform: none;\n  background: var(--widget-color);\n  border-radius: var(--widget-border-radius);\n  border: none;\n}\n@media (hover: hover) {\n  .lil-gui button:hover {\n    background: var(--hover-color);\n  }\n  .lil-gui button:focus {\n    box-shadow: inset 0 0 0 1px var(--focus-color);\n  }\n}\n.lil-gui button:active {\n  background: var(--focus-color);\n}\n\n@font-face {\n  font-family: \"lil-gui\";\n  src: url(\"data:application/font-woff;charset=utf-8;base64,d09GRgABAAAAAAUsAAsAAAAACJwAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAABHU1VCAAABCAAAAH4AAADAImwmYE9TLzIAAAGIAAAAPwAAAGBKqH5SY21hcAAAAcgAAAD0AAACrukyyJBnbHlmAAACvAAAAF8AAACEIZpWH2hlYWQAAAMcAAAAJwAAADZfcj2zaGhlYQAAA0QAAAAYAAAAJAC5AHhobXR4AAADXAAAABAAAABMAZAAAGxvY2EAAANsAAAAFAAAACgCEgIybWF4cAAAA4AAAAAeAAAAIAEfABJuYW1lAAADoAAAASIAAAIK9SUU/XBvc3QAAATEAAAAZgAAAJCTcMc2eJxVjbEOgjAURU+hFRBK1dGRL+ALnAiToyMLEzFpnPz/eAshwSa97517c/MwwJmeB9kwPl+0cf5+uGPZXsqPu4nvZabcSZldZ6kfyWnomFY/eScKqZNWupKJO6kXN3K9uCVoL7iInPr1X5baXs3tjuMqCtzEuagm/AAlzQgPAAB4nGNgYRBlnMDAysDAYM/gBiT5oLQBAwuDJAMDEwMrMwNWEJDmmsJwgCFeXZghBcjlZMgFCzOiKOIFAB71Bb8AeJy1kjFuwkAQRZ+DwRAwBtNQRUGKQ8OdKCAWUhAgKLhIuAsVSpWz5Bbkj3dEgYiUIszqWdpZe+Z7/wB1oCYmIoboiwiLT2WjKl/jscrHfGg/pKdMkyklC5Zs2LEfHYpjcRoPzme9MWWmk3dWbK9ObkWkikOetJ554fWyoEsmdSlt+uR0pCJR34b6t/TVg1SY3sYvdf8vuiKrpyaDXDISiegp17p7579Gp3p++y7HPAiY9pmTibljrr85qSidtlg4+l25GLCaS8e6rRxNBmsnERunKbaOObRz7N72ju5vdAjYpBXHgJylOAVsMseDAPEP8LYoUHicY2BiAAEfhiAGJgZWBgZ7RnFRdnVJELCQlBSRlATJMoLV2DK4glSYs6ubq5vbKrJLSbGrgEmovDuDJVhe3VzcXFwNLCOILB/C4IuQ1xTn5FPilBTj5FPmBAB4WwoqAHicY2BkYGAA4sk1sR/j+W2+MnAzpDBgAyEMQUCSg4EJxAEAwUgFHgB4nGNgZGBgSGFggJMhDIwMqEAYAByHATJ4nGNgAIIUNEwmAABl3AGReJxjYAACIQYlBiMGJ3wQAEcQBEV4nGNgZGBgEGZgY2BiAAEQyQWEDAz/wXwGAAsPATIAAHicXdBNSsNAHAXwl35iA0UQXYnMShfS9GPZA7T7LgIu03SSpkwzYTIt1BN4Ak/gKTyAeCxfw39jZkjymzcvAwmAW/wgwHUEGDb36+jQQ3GXGot79L24jxCP4gHzF/EIr4jEIe7wxhOC3g2TMYy4Q7+Lu/SHuEd/ivt4wJd4wPxbPEKMX3GI5+DJFGaSn4qNzk8mcbKSR6xdXdhSzaOZJGtdapd4vVPbi6rP+cL7TGXOHtXKll4bY1Xl7EGnPtp7Xy2n00zyKLVHfkHBa4IcJ2oD3cgggWvt/V/FbDrUlEUJhTn/0azVWbNTNr0Ens8de1tceK9xZmfB1CPjOmPH4kitmvOubcNpmVTN3oFJyjzCvnmrwhJTzqzVj9jiSX911FjeAAB4nG3HMRKCMBBA0f0giiKi4DU8k0V2GWbIZDOh4PoWWvq6J5V8If9NVNQcaDhyouXMhY4rPTcG7jwYmXhKq8Wz+p762aNaeYXom2n3m2dLTVgsrCgFJ7OTmIkYbwIbC6vIB7WmFfAAAA==\") format(\"woff\");\n}`;\n\nfunction _injectStyles( cssContent ) {\n\tconst injected = document.createElement( 'style' );\n\tinjected.innerHTML = cssContent;\n\tconst before = document.querySelector( 'head link[rel=stylesheet], head style' );\n\tif ( before ) {\n\t\tdocument.head.insertBefore( injected, before );\n\t} else {\n\t\tdocument.head.appendChild( injected );\n\t}\n}\n\nlet stylesInjected = false;\n\nclass GUI {\n\n\t/**\n\t * Creates a panel that holds controllers.\n\t * @example\n\t * new GUI();\n\t * new GUI( { container: document.getElementById( 'custom' ) } );\n\t *\n\t * @param {object} [options]\n\t * @param {boolean} [options.autoPlace=true]\n\t * Adds the GUI to `document.body` and fixes it to the top right of the page.\n\t *\n\t * @param {HTMLElement} [options.container]\n\t * Adds the GUI to this DOM element. Overrides `autoPlace`.\n\t *\n\t * @param {number} [options.width=245]\n\t * Width of the GUI in pixels, usually set when name labels become too long. Note that you can make\n\t * name labels wider in CSS with `.lil‑gui { ‑‑name‑width: 55% }`.\n\t *\n\t * @param {string} [options.title=Controls]\n\t * Name to display in the title bar.\n\t *\n\t * @param {boolean} [options.closeFolders=false]\n\t * Pass `true` to close all folders in this GUI by default.\n\t *\n\t * @param {boolean} [options.injectStyles=true]\n\t * Injects the default stylesheet into the page if this is the first GUI.\n\t * Pass `false` to use your own stylesheet.\n\t *\n\t * @param {number} [options.touchStyles=true]\n\t * Makes controllers larger on touch devices. Pass `false` to disable touch styles.\n\t *\n\t * @param {GUI} [options.parent]\n\t * Adds this GUI as a child in another GUI. Usually this is done for you by `addFolder()`.\n\t *\n\t */\n\tconstructor( {\n\t\tparent,\n\t\tautoPlace = parent === undefined,\n\t\tcontainer,\n\t\twidth,\n\t\ttitle = 'Controls',\n\t\tcloseFolders = false,\n\t\tinjectStyles = true,\n\t\ttouchStyles = true\n\t} = {} ) {\n\n\t\t/**\n\t\t * The GUI containing this folder, or `undefined` if this is the root GUI.\n\t\t * @type {GUI}\n\t\t */\n\t\tthis.parent = parent;\n\n\t\t/**\n\t\t * The top level GUI containing this folder, or `this` if this is the root GUI.\n\t\t * @type {GUI}\n\t\t */\n\t\tthis.root = parent ? parent.root : this;\n\n\t\t/**\n\t\t * The list of controllers and folders contained by this GUI.\n\t\t * @type {Array<GUI|Controller>}\n\t\t */\n\t\tthis.children = [];\n\n\t\t/**\n\t\t * The list of controllers contained by this GUI.\n\t\t * @type {Array<Controller>}\n\t\t */\n\t\tthis.controllers = [];\n\n\t\t/**\n\t\t * The list of folders contained by this GUI.\n\t\t * @type {Array<GUI>}\n\t\t */\n\t\tthis.folders = [];\n\n\t\t/**\n\t\t * Used to determine if the GUI is closed. Use `gui.open()` or `gui.close()` to change this.\n\t\t * @type {boolean}\n\t\t */\n\t\tthis._closed = false;\n\n\t\t/**\n\t\t * Used to determine if the GUI is hidden. Use `gui.show()` or `gui.hide()` to change this.\n\t\t * @type {boolean}\n\t\t */\n\t\tthis._hidden = false;\n\n\t\t/**\n\t\t * The outermost container element.\n\t\t * @type {HTMLElement}\n\t\t */\n\t\tthis.domElement = document.createElement( 'div' );\n\t\tthis.domElement.classList.add( 'lil-gui' );\n\n\t\t/**\n\t\t * The DOM element that contains the title.\n\t\t * @type {HTMLElement}\n\t\t */\n\t\tthis.$title = document.createElement( 'div' );\n\t\tthis.$title.classList.add( 'title' );\n\t\tthis.$title.setAttribute( 'role', 'button' );\n\t\tthis.$title.setAttribute( 'aria-expanded', true );\n\t\tthis.$title.setAttribute( 'tabindex', 0 );\n\n\t\tthis.$title.addEventListener( 'click', () => this.openAnimated( this._closed ) );\n\t\tthis.$title.addEventListener( 'keydown', e => {\n\t\t\tif ( e.code === 'Enter' || e.code === 'Space' ) {\n\t\t\t\te.preventDefault();\n\t\t\t\tthis.$title.click();\n\t\t\t}\n\t\t} );\n\n\t\t// enables :active pseudo class on mobile\n\t\tthis.$title.addEventListener( 'touchstart', () => {}, { passive: true } );\n\n\t\t/**\n\t\t * The DOM element that contains children.\n\t\t * @type {HTMLElement}\n\t\t */\n\t\tthis.$children = document.createElement( 'div' );\n\t\tthis.$children.classList.add( 'children' );\n\n\t\tthis.domElement.appendChild( this.$title );\n\t\tthis.domElement.appendChild( this.$children );\n\n\t\tthis.title( title );\n\n\t\tif ( this.parent ) {\n\n\t\t\tthis.parent.children.push( this );\n\t\t\tthis.parent.folders.push( this );\n\n\t\t\tthis.parent.$children.appendChild( this.domElement );\n\n\t\t\t// Stop the constructor early, everything onward only applies to root GUI's\n\t\t\treturn;\n\n\t\t}\n\n\t\tthis.domElement.classList.add( 'root' );\n\n\t\tif ( touchStyles ) {\n\t\t\tthis.domElement.classList.add( 'allow-touch-styles' );\n\t\t}\n\n\t\t// Inject stylesheet if we haven't done that yet\n\t\tif ( !stylesInjected && injectStyles ) {\n\t\t\t_injectStyles( stylesheet );\n\t\t\tstylesInjected = true;\n\t\t}\n\n\t\tif ( container ) {\n\n\t\t\tcontainer.appendChild( this.domElement );\n\n\t\t} else if ( autoPlace ) {\n\n\t\t\tthis.domElement.classList.add( 'autoPlace' );\n\t\t\tdocument.body.appendChild( this.domElement );\n\n\t\t}\n\n\t\tif ( width ) {\n\t\t\tthis.domElement.style.setProperty( '--width', width + 'px' );\n\t\t}\n\n\t\tthis._closeFolders = closeFolders;\n\n\t}\n\n\t/**\n\t * Adds a controller to the GUI, inferring controller type using the `typeof` operator.\n\t * @example\n\t * gui.add( object, 'property' );\n\t * gui.add( object, 'number', 0, 100, 1 );\n\t * gui.add( object, 'options', [ 1, 2, 3 ] );\n\t *\n\t * @param {object} object The object the controller will modify.\n\t * @param {string} property Name of the property to control.\n\t * @param {number|object|Array} [$1] Minimum value for number controllers, or the set of\n\t * selectable values for a dropdown.\n\t * @param {number} [max] Maximum value for number controllers.\n\t * @param {number} [step] Step value for number controllers.\n\t * @returns {Controller}\n\t */\n\tadd( object, property, $1, max, step ) {\n\n\t\tif ( Object( $1 ) === $1 ) {\n\n\t\t\treturn new OptionController( this, object, property, $1 );\n\n\t\t}\n\n\t\tconst initialValue = object[ property ];\n\n\t\tswitch ( typeof initialValue ) {\n\n\t\t\tcase 'number':\n\n\t\t\t\treturn new NumberController( this, object, property, $1, max, step );\n\n\t\t\tcase 'boolean':\n\n\t\t\t\treturn new BooleanController( this, object, property );\n\n\t\t\tcase 'string':\n\n\t\t\t\treturn new StringController( this, object, property );\n\n\t\t\tcase 'function':\n\n\t\t\t\treturn new FunctionController( this, object, property );\n\n\t\t}\n\n\t\tconsole.error( `gui.add failed\n\tproperty:`, property, `\n\tobject:`, object, `\n\tvalue:`, initialValue );\n\n\t}\n\n\t/**\n\t * Adds a color controller to the GUI.\n\t * @example\n\t * params = {\n\t * \tcssColor: '#ff00ff',\n\t * \trgbColor: { r: 0, g: 0.2, b: 0.4 },\n\t * \tcustomRange: [ 0, 127, 255 ],\n\t * };\n\t *\n\t * gui.addColor( params, 'cssColor' );\n\t * gui.addColor( params, 'rgbColor' );\n\t * gui.addColor( params, 'customRange', 255 );\n\t *\n\t * @param {object} object The object the controller will modify.\n\t * @param {string} property Name of the property to control.\n\t * @param {number} rgbScale Maximum value for a color channel when using an RGB color. You may\n\t * need to set this to 255 if your colors are too bright.\n\t * @returns {Controller}\n\t */\n\taddColor( object, property, rgbScale = 1 ) {\n\t\treturn new ColorController( this, object, property, rgbScale );\n\t}\n\n\t/**\n\t * Adds a folder to the GUI, which is just another GUI. This method returns\n\t * the nested GUI so you can add controllers to it.\n\t * @example\n\t * const folder = gui.addFolder( 'Position' );\n\t * folder.add( position, 'x' );\n\t * folder.add( position, 'y' );\n\t * folder.add( position, 'z' );\n\t *\n\t * @param {string} title Name to display in the folder's title bar.\n\t * @returns {GUI}\n\t */\n\taddFolder( title ) {\n\t\tconst folder = new GUI( { parent: this, title } );\n\t\tif ( this.root._closeFolders ) folder.close();\n\t\treturn folder;\n\t}\n\n\t/**\n\t * Recalls values that were saved with `gui.save()`.\n\t * @param {object} obj\n\t * @param {boolean} recursive Pass false to exclude folders descending from this GUI.\n\t * @returns {this}\n\t */\n\tload( obj, recursive = true ) {\n\n\t\tif ( obj.controllers ) {\n\n\t\t\tthis.controllers.forEach( c => {\n\n\t\t\t\tif ( c instanceof FunctionController ) return;\n\n\t\t\t\tif ( c._name in obj.controllers ) {\n\t\t\t\t\tc.load( obj.controllers[ c._name ] );\n\t\t\t\t}\n\n\t\t\t} );\n\n\t\t}\n\n\t\tif ( recursive && obj.folders ) {\n\n\t\t\tthis.folders.forEach( f => {\n\n\t\t\t\tif ( f._title in obj.folders ) {\n\t\t\t\t\tf.load( obj.folders[ f._title ] );\n\t\t\t\t}\n\n\t\t\t} );\n\n\t\t}\n\n\t\treturn this;\n\n\t}\n\n\t/**\n\t * Returns an object mapping controller names to values. The object can be passed to `gui.load()` to\n\t * recall these values.\n\t * @example\n\t * {\n\t * \tcontrollers: {\n\t * \t\tprop1: 1,\n\t * \t\tprop2: 'value',\n\t * \t\t...\n\t * \t},\n\t * \tfolders: {\n\t * \t\tfolderName1: { controllers, folders },\n\t * \t\tfolderName2: { controllers, folders }\n\t * \t\t...\n\t * \t}\n\t * }\n\t *\n\t * @param {boolean} recursive Pass false to exclude folders descending from this GUI.\n\t * @returns {object}\n\t */\n\tsave( recursive = true ) {\n\n\t\tconst obj = {\n\t\t\tcontrollers: {},\n\t\t\tfolders: {}\n\t\t};\n\n\t\tthis.controllers.forEach( c => {\n\n\t\t\tif ( c instanceof FunctionController ) return;\n\n\t\t\tif ( c._name in obj.controllers ) {\n\t\t\t\tthrow new Error( `Cannot save GUI with duplicate property \"${c._name}\"` );\n\t\t\t}\n\n\t\t\tobj.controllers[ c._name ] = c.save();\n\n\t\t} );\n\n\t\tif ( recursive ) {\n\n\t\t\tthis.folders.forEach( f => {\n\n\t\t\t\tif ( f._title in obj.folders ) {\n\t\t\t\t\tthrow new Error( `Cannot save GUI with duplicate folder \"${f._title}\"` );\n\t\t\t\t}\n\n\t\t\t\tobj.folders[ f._title ] = f.save();\n\n\t\t\t} );\n\n\t\t}\n\n\t\treturn obj;\n\n\t}\n\n\t/**\n\t * Opens a GUI or folder. GUI and folders are open by default.\n\t * @param {boolean} open Pass false to close.\n\t * @returns {this}\n\t * @example\n\t * gui.open(); // open\n\t * gui.open( false ); // close\n\t * gui.open( gui._closed ); // toggle\n\t */\n\topen( open = true ) {\n\n\t\tthis._setClosed( !open );\n\n\t\tthis.$title.setAttribute( 'aria-expanded', !this._closed );\n\t\tthis.domElement.classList.toggle( 'closed', this._closed );\n\n\t\treturn this;\n\n\t}\n\n\t/**\n\t * Closes the GUI.\n\t * @returns {this}\n\t */\n\tclose() {\n\t\treturn this.open( false );\n\t}\n\n\t_setClosed( closed ) {\n\t\tif ( this._closed === closed ) return;\n\t\tthis._closed = closed;\n\t\tthis._callOnOpenClose( this );\n\t}\n\n\t/**\n\t * Shows the GUI after it's been hidden.\n\t * @param {boolean} show\n\t * @returns {this}\n\t * @example\n\t * gui.show();\n\t * gui.show( false ); // hide\n\t * gui.show( gui._hidden ); // toggle\n\t */\n\tshow( show = true ) {\n\n\t\tthis._hidden = !show;\n\n\t\tthis.domElement.style.display = this._hidden ? 'none' : '';\n\n\t\treturn this;\n\n\t}\n\n\t/**\n\t * Hides the GUI.\n\t * @returns {this}\n\t */\n\thide() {\n\t\treturn this.show( false );\n\t}\n\n\topenAnimated( open = true ) {\n\n\t\t// set state immediately\n\t\tthis._setClosed( !open );\n\n\t\tthis.$title.setAttribute( 'aria-expanded', !this._closed );\n\n\t\t// wait for next frame to measure $children\n\t\trequestAnimationFrame( () => {\n\n\t\t\t// explicitly set initial height for transition\n\t\t\tconst initialHeight = this.$children.clientHeight;\n\t\t\tthis.$children.style.height = initialHeight + 'px';\n\n\t\t\tthis.domElement.classList.add( 'transition' );\n\n\t\t\tconst onTransitionEnd = e => {\n\t\t\t\tif ( e.target !== this.$children ) return;\n\t\t\t\tthis.$children.style.height = '';\n\t\t\t\tthis.domElement.classList.remove( 'transition' );\n\t\t\t\tthis.$children.removeEventListener( 'transitionend', onTransitionEnd );\n\t\t\t};\n\n\t\t\tthis.$children.addEventListener( 'transitionend', onTransitionEnd );\n\n\t\t\t// todo: this is wrong if children's scrollHeight makes for a gui taller than maxHeight\n\t\t\tconst targetHeight = !open ? 0 : this.$children.scrollHeight;\n\n\t\t\tthis.domElement.classList.toggle( 'closed', !open );\n\n\t\t\trequestAnimationFrame( () => {\n\t\t\t\tthis.$children.style.height = targetHeight + 'px';\n\t\t\t} );\n\n\t\t} );\n\n\t\treturn this;\n\n\t}\n\n\t/**\n\t * Change the title of this GUI.\n\t * @param {string} title\n\t * @returns {this}\n\t */\n\ttitle( title ) {\n\t\t/**\n\t\t * Current title of the GUI. Use `gui.title( 'Title' )` to modify this value.\n\t\t * @type {string}\n\t\t */\n\t\tthis._title = title;\n\t\tthis.$title.textContent = title;\n\t\treturn this;\n\t}\n\n\t/**\n\t * Resets all controllers to their initial values.\n\t * @param {boolean} recursive Pass false to exclude folders descending from this GUI.\n\t * @returns {this}\n\t */\n\treset( recursive = true ) {\n\t\tconst controllers = recursive ? this.controllersRecursive() : this.controllers;\n\t\tcontrollers.forEach( c => c.reset() );\n\t\treturn this;\n\t}\n\n\t/**\n\t * Pass a function to be called whenever a controller in this GUI changes.\n\t * @param {function({object:object, property:string, value:any, controller:Controller})} callback\n\t * @returns {this}\n\t * @example\n\t * gui.onChange( event => {\n\t * \tevent.object     // object that was modified\n\t * \tevent.property   // string, name of property\n\t * \tevent.value      // new value of controller\n\t * \tevent.controller // controller that was modified\n\t * } );\n\t */\n\tonChange( callback ) {\n\t\t/**\n\t\t * Used to access the function bound to `onChange` events. Don't modify this value\n\t\t * directly. Use the `gui.onChange( callback )` method instead.\n\t\t * @type {Function}\n\t\t */\n\t\tthis._onChange = callback;\n\t\treturn this;\n\t}\n\n\t_callOnChange( controller ) {\n\n\t\tif ( this.parent ) {\n\t\t\tthis.parent._callOnChange( controller );\n\t\t}\n\n\t\tif ( this._onChange !== undefined ) {\n\t\t\tthis._onChange.call( this, {\n\t\t\t\tobject: controller.object,\n\t\t\t\tproperty: controller.property,\n\t\t\t\tvalue: controller.getValue(),\n\t\t\t\tcontroller\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Pass a function to be called whenever a controller in this GUI has finished changing.\n\t * @param {function({object:object, property:string, value:any, controller:Controller})} callback\n\t * @returns {this}\n\t * @example\n\t * gui.onFinishChange( event => {\n\t * \tevent.object     // object that was modified\n\t * \tevent.property   // string, name of property\n\t * \tevent.value      // new value of controller\n\t * \tevent.controller // controller that was modified\n\t * } );\n\t */\n\tonFinishChange( callback ) {\n\t\t/**\n\t\t * Used to access the function bound to `onFinishChange` events. Don't modify this value\n\t\t * directly. Use the `gui.onFinishChange( callback )` method instead.\n\t\t * @type {Function}\n\t\t */\n\t\tthis._onFinishChange = callback;\n\t\treturn this;\n\t}\n\n\t_callOnFinishChange( controller ) {\n\n\t\tif ( this.parent ) {\n\t\t\tthis.parent._callOnFinishChange( controller );\n\t\t}\n\n\t\tif ( this._onFinishChange !== undefined ) {\n\t\t\tthis._onFinishChange.call( this, {\n\t\t\t\tobject: controller.object,\n\t\t\t\tproperty: controller.property,\n\t\t\t\tvalue: controller.getValue(),\n\t\t\t\tcontroller\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Pass a function to be called when this GUI or its descendants are opened or closed.\n\t * @param {function(GUI)} callback\n\t * @returns {this}\n\t * @example\n\t * gui.onOpenClose( changedGUI => {\n\t * \tconsole.log( changedGUI._closed );\n\t * } );\n\t */\n\tonOpenClose( callback ) {\n\t\tthis._onOpenClose = callback;\n\t\treturn this;\n\t}\n\n\t_callOnOpenClose( changedGUI ) {\n\t\tif ( this.parent ) {\n\t\t\tthis.parent._callOnOpenClose( changedGUI );\n\t\t}\n\n\t\tif ( this._onOpenClose !== undefined ) {\n\t\t\tthis._onOpenClose.call( this, changedGUI );\n\t\t}\n\t}\n\n\t/**\n\t * Destroys all DOM elements and event listeners associated with this GUI.\n\t */\n\tdestroy() {\n\n\t\tif ( this.parent ) {\n\t\t\tthis.parent.children.splice( this.parent.children.indexOf( this ), 1 );\n\t\t\tthis.parent.folders.splice( this.parent.folders.indexOf( this ), 1 );\n\t\t}\n\n\t\tif ( this.domElement.parentElement ) {\n\t\t\tthis.domElement.parentElement.removeChild( this.domElement );\n\t\t}\n\n\t\tArray.from( this.children ).forEach( c => c.destroy() );\n\n\t}\n\n\t/**\n\t * Returns an array of controllers contained by this GUI and its descendents.\n\t * @returns {Controller[]}\n\t */\n\tcontrollersRecursive() {\n\t\tlet controllers = Array.from( this.controllers );\n\t\tthis.folders.forEach( f => {\n\t\t\tcontrollers = controllers.concat( f.controllersRecursive() );\n\t\t} );\n\t\treturn controllers;\n\t}\n\n\t/**\n\t * Returns an array of folders contained by this GUI and its descendents.\n\t * @returns {GUI[]}\n\t */\n\tfoldersRecursive() {\n\t\tlet folders = Array.from( this.folders );\n\t\tthis.folders.forEach( f => {\n\t\t\tfolders = folders.concat( f.foldersRecursive() );\n\t\t} );\n\t\treturn folders;\n\t}\n\n}\n\nexport default GUI;\nexport { BooleanController, ColorController, Controller, FunctionController, GUI, NumberController, OptionController, StringController };\n","import GUI from 'lil-gui';\r\nimport Plotly, { type Data, type Layout } from 'plotly.js-dist';\r\nimport * as THREE from 'three';\r\nimport { OrbitControls, ViewHelper } from 'three/examples/jsm/Addons';\r\nimport Stats from 'three/examples/jsm/libs/stats.module';\r\nimport { type Simulation } from '../Simulation';\r\nimport { type State } from '../State';\r\nimport { type Universe } from '../Universe';\r\nimport { type Visualizer } from '../Visualizer';\r\n\r\nlet animationId: number | null = null;\r\n\r\n/**\r\n * Clips a number to a minimum and maximum value.\r\n * @param x number to clip.\r\n * @param min minimum value.\r\n * @param max maximum value.\r\n * @returns clipped value.\r\n */\r\nfunction clipMinMax(x: number, min: number, max: number): number {\r\n  if (x < min) return min;\r\n  if (x > max) return max;\r\n  return x;\r\n}\r\n\r\n/**\r\n * Container object for body trails in a 2D universe based in Plotly.\r\n */\r\nclass PlotlyUniverseTrail {\r\n  data: {\r\n    x: (number | null)[];\r\n    y: (number | null)[];\r\n    mode: 'markers';\r\n    marker: {\r\n      size: number;\r\n      color: string;\r\n    };\r\n  } = {\r\n      x: [],\r\n      y: [],\r\n      mode: 'markers',\r\n      marker: {\r\n        size: 1,\r\n        color: 'white',\r\n      },\r\n    };\r\n  trailLength: number;\r\n  maxTrailLength: number;\r\n  trailInd: number;\r\n\r\n  /**\r\n   * Constructor for PlotlyUniverseTrail\r\n   * @param maxTrailLength max number of trail points to keep.\r\n   * @param color color of the trail.\r\n   */\r\n  constructor(maxTrailLength: number, color: string) {\r\n    this.data.marker.color = color;\r\n    this.trailLength = 0;\r\n    this.maxTrailLength = maxTrailLength;\r\n    this.trailInd = 0;\r\n  }\r\n\r\n  /**\r\n   * Add a trail point to the trail data, or replace an existing trail point if the max trail length has been reached.\r\n   * @param x x position.\r\n   * @param y y position.\r\n   */\r\n  addTrail(x: number, y: number): void {\r\n    if (this.trailLength < this.maxTrailLength) {\r\n      this.data.x.push(x);\r\n      this.data.y.push(y);\r\n      this.trailLength++;\r\n    } else {\r\n      this.data.x[this.trailInd] = x;\r\n      this.data.y[this.trailInd] = y;\r\n      this.trailInd = (this.trailInd + 1) % this.trailLength;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Pop all trail points from the trail data.\r\n   */\r\n  popAllTrails(): void {\r\n    this.data.x = [];\r\n    this.data.y = [];\r\n    this.trailLength = 0;\r\n    this.trailInd = 0;\r\n  }\r\n}\r\n\r\n/**\r\n * 2D real-time visualizer using Plotly.\r\n * @category Visualizers\r\n */\r\nexport class RealTimeVisualizer implements Visualizer {\r\n  simulation: Simulation;\r\n  divId: string = '';\r\n  universeTrails: PlotlyUniverseTrail[] = [];\r\n\r\n  /**\r\n   * Constructor for RealTimeVisualizer\r\n   * @param simulation simulation object\r\n   */\r\n  constructor(simulation: Simulation) {\r\n    this.simulation = simulation;\r\n  }\r\n\r\n  /**\r\n   * Adds default controls using lil-gui to the visualization.\r\n   * @param parentElement parent element to place the controller div in.\r\n   */\r\n  private addControls(parentElement: HTMLElement) {\r\n    const gui = new GUI({\r\n      container: parentElement,\r\n    });\r\n    gui.domElement.style.position = 'absolute';\r\n    gui.domElement.style.top = '0';\r\n    gui.domElement.style.left = '0';\r\n    gui.domElement.style.zIndex = '1000';\r\n\r\n    const config = this.simulation.controls;\r\n    gui.add(config, 'speed');\r\n    gui.add(config, 'showTrails')\r\n      .onChange((value: boolean) => {\r\n        if (value === false) {\r\n          this.universeTrails.forEach((ut) => ut.popAllTrails());\r\n        }\r\n        config.showTrails = value;\r\n      });\r\n    const showUniverseFolder = gui.addFolder('Show Universe');\r\n    showUniverseFolder.open(false);\r\n    this.simulation.universes.forEach((u, i) => {\r\n      showUniverseFolder\r\n        .add(config.showUniverse, u.label)\r\n        .onChange((value: boolean) => {\r\n          if (value === false) {\r\n            this.universeTrails[i].popAllTrails();\r\n          }\r\n          config.showUniverse[u.label] = value;\r\n        });\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Simulate and play the visualization.\r\n   * @param divId div id to render the visualization in.\r\n   * @param width width of the visualization.\r\n   * @param height height of the visualization.\r\n   */\r\n  start(divId: string, width: number, height: number): void {\r\n    if (this.divId !== '') {\r\n      // throw new Error(\r\n      //   'Simulation already playing. Stop the current playtime before initiating a new one.',\r\n      // );\r\n      console.error('Simulation already playing. Stop the current playtime before initiating a new one.');\r\n      return;\r\n    }\r\n    this.divId = divId;\r\n    let element = document.getElementById(divId);\r\n    if (element === null) {\r\n      return;\r\n    }\r\n    // const width = element.clientWidth;\r\n    // const height = element.clientHeight;\r\n    let maxWidth = 0;\r\n    let maxHeight = 0;\r\n    this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => {\r\n      maxWidth = Math.max(maxWidth, Math.abs(b.position.x));\r\n      maxHeight = Math.max(maxHeight, Math.abs(b.position.y));\r\n    }));\r\n    const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth);\r\n\r\n    const layout: Partial<Layout> = {\r\n      paper_bgcolor: '#000000',\r\n      plot_bgcolor: '#000000',\r\n      font: {\r\n        color: '#bfbfbf',\r\n      },\r\n      xaxis: {\r\n        autorange: false,\r\n        range: [-(width / 2) / scale, width / 2 / scale],\r\n      },\r\n      yaxis: {\r\n        autorange: false,\r\n        range: [-(height / 2) / scale, height / 2 / scale],\r\n      },\r\n      // uirevision: 'true',\r\n      showlegend: false,\r\n    };\r\n\r\n    if (this.simulation.controller === 'ui') {\r\n      this.addControls(element);\r\n    }\r\n\r\n    let stats: Stats | undefined;\r\n    if (this.simulation.showDebugInfo) {\r\n      stats = new Stats();\r\n      stats.dom.style.position = 'absolute';\r\n      stats.dom.style.bottom = '0px';\r\n      stats.dom.style.removeProperty('top');\r\n      element.appendChild(stats.dom);\r\n    }\r\n\r\n    const init_data: Data[] = this.simulation.universes.flatMap(\r\n      (uni: Universe): Data[] => {\r\n        const currTrail = new PlotlyUniverseTrail(\r\n          this.simulation.getMaxTrailLength(),\r\n          typeof uni.color === 'string' ? uni.color : uni.color[0],\r\n        );\r\n        this.universeTrails.push(currTrail);\r\n        const currData: Data = {\r\n          x: uni.currState.bodies.map((body) => body.position.x),\r\n          y: uni.currState.bodies.map((body) => body.position.y),\r\n          type: 'scatter',\r\n          mode: 'markers',\r\n          marker: {\r\n            color: uni.color,\r\n            sizemin: 6,\r\n            size: uni.currState.bodies.map((body) => Math.min(10, body.mass)),\r\n          },\r\n        };\r\n        if (this.simulation.getShowTrails()) {\r\n          uni.currState.bodies.forEach((b) => {\r\n            currTrail.addTrail(b.position.x, b.position.y);\r\n          });\r\n          return [currData, currTrail.data];\r\n        }\r\n        return [\r\n          currData,\r\n          {\r\n            x: [],\r\n            y: [],\r\n          },\r\n        ];\r\n      },\r\n    );\r\n\r\n    Plotly.newPlot(divId, init_data, layout, {\r\n      scrollZoom: true,\r\n      modeBarButtonsToRemove: [\r\n        'lasso2d',\r\n        'select2d',\r\n        'toImage',\r\n        'resetScale2d',\r\n      ],\r\n    });\r\n\r\n    const timePerFrame = 1000 / this.simulation.maxFrameRate;\r\n    if (animationId !== null) return;\r\n    let lastPaintTimestampMs = 0;\r\n    let lastSimTimestampMs = 0;\r\n\r\n    /**\r\n     * Simulate a step in the simulation\r\n     * @param timestampMs current timestamp in milliseconds, sourced from requestAnimationFrame\r\n     */\r\n    const step = (timestampMs: number): void => {\r\n      this.simulation.simulateStep(\r\n        (this.simulation.controls.speed\r\n          * Math.min(timestampMs - lastSimTimestampMs, 33.33))\r\n          / 1000,\r\n      );\r\n      lastSimTimestampMs = timestampMs;\r\n    };\r\n\r\n    /**\r\n     * Paint the visualization\r\n     * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame\r\n     */\r\n    const paint = (timestampMs: number) => {\r\n      if (\r\n        this.simulation.controls.speed === 0\r\n        || this.simulation.controls.paused\r\n      ) {\r\n        animationId = requestAnimationFrame(paint);\r\n        return;\r\n      }\r\n      step(timestampMs);\r\n\r\n      if (\r\n        timePerFrame > 0\r\n        && timestampMs - lastPaintTimestampMs < timePerFrame\r\n      ) {\r\n        animationId = requestAnimationFrame(paint);\r\n        return;\r\n      }\r\n      lastPaintTimestampMs = timestampMs;\r\n\r\n      const new_data = this.simulation.universes.flatMap(\r\n        (uni: Universe, i: number): Data[] => {\r\n          if (!this.simulation.getShowUniverse(uni.label)) {\r\n            return [\r\n              {\r\n                x: [],\r\n                y: [],\r\n              },\r\n              {},\r\n            ];\r\n          }\r\n          const currData: Data = {\r\n            x: uni.currState.bodies.map((body) => body.position.x),\r\n            y: uni.currState.bodies.map((body) => body.position.y),\r\n            hovertext: uni.currState.bodies.map((body) => body.label),\r\n            marker: {\r\n              size: uni.currState.bodies.map((body) => Math.min(10, body.mass)),\r\n              color: uni.color,\r\n              sizemin: 6,\r\n            },\r\n            mode: 'markers',\r\n          };\r\n\r\n          let trailData: Data = {};\r\n          if (this.simulation.getShowTrails()) {\r\n            const currTrail = this.universeTrails[i];\r\n            uni.currState.bodies.forEach((b) => {\r\n              currTrail.addTrail(b.position.x, b.position.y);\r\n            });\r\n            trailData = currTrail.data;\r\n          }\r\n          return [currData, trailData];\r\n        },\r\n      );\r\n\r\n      Plotly.react(divId, new_data, layout);\r\n      if (this.simulation.showDebugInfo && stats) {\r\n        stats.update();\r\n      }\r\n      animationId = requestAnimationFrame(paint);\r\n    };\r\n\r\n    animationId = requestAnimationFrame(paint);\r\n  }\r\n\r\n  /**\r\n   * Stop the simulation and visualization.\r\n   */\r\n  stop(): void {\r\n    Plotly.purge(this.divId);\r\n    this.divId = '';\r\n    this.universeTrails.forEach((ut) => {\r\n      ut.popAllTrails();\r\n    });\r\n    this.universeTrails = [];\r\n  }\r\n}\r\n\r\n/**\r\n * Container object for body trails in a 3D universe based in Three.js.\r\n */\r\nclass ThreeUniverseTrail {\r\n  /**\r\n   * Singular Points object containing all trail points.\r\n   */\r\n  trails: THREE.Points;\r\n  trailInd: number;\r\n  trailLength: number;\r\n  maxTrailLength: number;\r\n\r\n  /**\r\n   * Constructor for ThreeUniverseTrail.\r\n   * @param maxTrailLength max number of trail points to keep.\r\n   * @param color color of the trace points.\r\n   * @param scene scene to add trail points object to.\r\n   * @param scale scale of the visualizationl, used to set the size of the trail point.\r\n   */\r\n  constructor(\r\n    maxTrailLength: number,\r\n    color: string,\r\n    scene: THREE.Scene,\r\n    scale: number,\r\n  ) {\r\n    const geometry = new THREE.BufferGeometry();\r\n    geometry.setAttribute(\r\n      'position',\r\n      new THREE.BufferAttribute(new Float32Array(0), 3),\r\n    );\r\n    this.trails = new THREE.Points(\r\n      geometry,\r\n      new THREE.PointsMaterial({\r\n        color,\r\n        size: 0.005 * scale,\r\n      }),\r\n    );\r\n    scene.add(this.trails);\r\n    this.trailInd = 0;\r\n    this.trailLength = 0;\r\n    this.maxTrailLength = maxTrailLength;\r\n  }\r\n\r\n  /**\r\n   * Add a trail point at the given position to the scene. Replace an existing trail point if the max trail length has been reached.\r\n   * @param pos position to add trace point at.\r\n   */\r\n  addTrail(pos: THREE.Vector3) {\r\n    if (this.trailLength < this.maxTrailLength) {\r\n      this.trails.visible = true;\r\n      this.trailLength++;\r\n      const posArray = new Float32Array(this.trailLength * 3);\r\n      posArray.set(this.trails.geometry.attributes.position.array);\r\n      posArray.set(pos.toArray(), this.trailLength * 3 - 3);\r\n      this.trails.geometry.setAttribute(\r\n        'position',\r\n        new THREE.BufferAttribute(posArray, 3),\r\n      );\r\n      this.trails.geometry.attributes.position.needsUpdate = true;\r\n    } else {\r\n      this.trails.geometry.attributes.position.array.set(\r\n        pos.toArray(),\r\n        this.trailInd * 3,\r\n      );\r\n      this.trailInd = (this.trailInd + 1) % this.maxTrailLength;\r\n      this.trails.geometry.attributes.position.needsUpdate = true;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Pop all trail points.\r\n   */\r\n  popAllTrails(): void {\r\n    this.trails.visible = false;\r\n    this.trails.geometry.setAttribute(\r\n      'position',\r\n      new THREE.BufferAttribute(new Float32Array(0), 3),\r\n    );\r\n    this.trailInd = 0;\r\n    this.trailLength = 0;\r\n  }\r\n}\r\n\r\n/**\r\n * 3D real-time visualizer using Three.js.\r\n * @category Visualizers\r\n */\r\nexport class RealTimeVisualizer3D implements Visualizer {\r\n  simulation: Simulation;\r\n  scene?: THREE.Scene;\r\n  universeTrails: ThreeUniverseTrail[] = [];\r\n\r\n  /**\r\n   * Constructor for RealTimeVisualizer3D.\r\n   * @param simulation simulation object.\r\n   */\r\n  constructor(simulation: Simulation) {\r\n    this.simulation = simulation;\r\n  }\r\n\r\n  /**\r\n   * Adds default controls to the visualization.\r\n   * @param parentElement parent element to place the controller div in.\r\n   */\r\n  private addControls(parentElement: HTMLElement) {\r\n    const gui = new GUI({\r\n      container: parentElement,\r\n    });\r\n    gui.domElement.style.position = 'absolute';\r\n    gui.domElement.style.top = '0';\r\n    gui.domElement.style.left = '0';\r\n    gui.domElement.style.zIndex = '1000';\r\n\r\n    const config = this.simulation.controls;\r\n    gui.add(config, 'speed');\r\n    gui.add(config, 'showTrails')\r\n      .onChange((value: boolean) => {\r\n        if (value === false) {\r\n          this.universeTrails.forEach((ut) => {\r\n            ut.popAllTrails();\r\n          });\r\n        }\r\n        config.showTrails = value;\r\n      });\r\n    const showUniverseFolder = gui.addFolder('Show Universe');\r\n    showUniverseFolder.open(false);\r\n    this.simulation.universes.forEach((u, i) => {\r\n      showUniverseFolder\r\n        .add(config.showUniverse, u.label)\r\n        .onChange((value: boolean) => {\r\n          if (value === false) {\r\n            this.universeTrails[i].popAllTrails();\r\n          }\r\n          config.showUniverse[u.label] = value;\r\n        });\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Simulate and play the visualization\r\n   * @param divId div id to render the visualization in\r\n   * @param width width of the visualization.\r\n   * @param height height of the visualization.\r\n   */\r\n  start(divId: string, width: number, height: number): void {\r\n    if (this.scene !== undefined) {\r\n      // throw new Error(\r\n      //   'Simulation already playing. Stop the current playtime before initiating a new one.',\r\n      // );\r\n      console.error('Simulation already playing. Stop the current playtime before initiating a new one.');\r\n      return;\r\n    }\r\n    let element = document.getElementById(divId);\r\n    if (element === null) {\r\n      return;\r\n    }\r\n    element.style.position = 'relative';\r\n    // const width = element.clientWidth;\r\n    // const height = element.clientHeight;\r\n    let maxWidth = 0;\r\n    let maxHeight = 0;\r\n    this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => {\r\n      maxWidth = Math.max(maxWidth, Math.abs(b.position.x));\r\n      maxHeight = Math.max(maxHeight, Math.abs(b.position.y));\r\n    }));\r\n    const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth);\r\n\r\n    this.scene = new THREE.Scene();\r\n\r\n    const camera = new THREE.OrthographicCamera(\r\n      width / -2,\r\n      width / 2,\r\n      height / 2,\r\n      height / -2,\r\n      0,\r\n      10000000000,\r\n    );\r\n    camera.position.set(0, 0, Math.max(width, height));\r\n\r\n    const renderer = new THREE.WebGLRenderer();\r\n    renderer.setSize(width, height);\r\n    renderer.autoClear = false;\r\n    element.appendChild(renderer.domElement);\r\n\r\n    let stats: Stats | undefined;\r\n    if (this.simulation.showDebugInfo) {\r\n      stats = new Stats();\r\n      stats.dom.style.position = 'absolute';\r\n      stats.dom.style.right = '0px';\r\n      stats.dom.style.removeProperty('left');\r\n      element.appendChild(stats.dom);\r\n    }\r\n\r\n    if (this.simulation.controller === 'ui') {\r\n      this.addControls(element);\r\n    }\r\n\r\n    // const earthDiv = document.createElement('div');\r\n    // earthDiv.className = 'label';\r\n    // earthDiv.textContent = 'Earthhgkjfdghkjfgh';\r\n    // earthDiv.style.backgroundColor = 'transparent';\r\n    // earthDiv.style.color = 'white';\r\n    // earthDiv.style.fontFamily = 'sans-serif';\r\n    // earthDiv.style.background = 'rgba(0, 0, 0, 0.6)';\r\n\r\n    // const earthLabel = new CSS2DObject(earthDiv);\r\n    // earthLabel.position.set(0, 0, 0);\r\n    // earthLabel.center.set(0, 1);\r\n    // // arr[0].add(earthLabel);\r\n    // earthLabel.layers.set(0);\r\n    // const labelRenderer = new CSS2DRenderer();\r\n    // labelRenderer.setSize(width, height);\r\n    // labelRenderer.domElement.style.position = 'absolute';\r\n    // labelRenderer.domElement.style.top = '0px';\r\n    // element.appendChild(labelRenderer.domElement);\r\n\r\n    const orbitControls = new OrbitControls(camera, renderer.domElement);\r\n    orbitControls.listenToKeyEvents(window);\r\n    orbitControls.update();\r\n\r\n    const axesHelper = new THREE.AxesHelper(width);\r\n    this.scene.add(axesHelper);\r\n    const viewHelper = new ViewHelper(camera, renderer.domElement);\r\n\r\n    // var m: Map<string, THREE.LineSegments> = new Map();\r\n    let arr: THREE.LineSegments[] = [];\r\n\r\n    this.simulation.universes.forEach((u) => {\r\n      this.universeTrails.push(\r\n        new ThreeUniverseTrail(\r\n          this.simulation.maxTrailLength,\r\n          typeof u.color === 'string' ? u.color : u.color[0],\r\n          this.scene!,\r\n          scale,\r\n        ),\r\n      );\r\n      u.currState.bodies.forEach((b) => {\r\n        const sph = new THREE.SphereGeometry(\r\n          clipMinMax(Math.log2(b.mass) - 70, 10, 40),\r\n          8,\r\n          8,\r\n        );\r\n        const curr = new THREE.WireframeGeometry(sph);\r\n        const line = new THREE.LineSegments(\r\n          curr,\r\n          new THREE.LineBasicMaterial({\r\n            // @ts-ignore\r\n            color: new THREE.Color(u.color),\r\n          }),\r\n        );\r\n        this.scene!.add(line);\r\n        line.position.copy(b.position.clone()\r\n          .multiplyScalar(scale));\r\n        // m.set(u.label + \" \" + b.label, line);\r\n        arr.push(line);\r\n      });\r\n    });\r\n    // arr[0].add(earthLabel)\r\n\r\n    const timePerFrame = 1000 / this.simulation.maxFrameRate;\r\n    let lastSimTimestampMs = performance.now();\r\n    let lastPaint = performance.now();\r\n\r\n    /**\r\n     * Simulate a step in the simulation\r\n     * @param timestampMs current timestamp in milliseconds, sourced from requestAnimationFrame\r\n     */\r\n    const step = (timestampMs: number) => {\r\n      this.simulation.simulateStep(\r\n        (this.simulation.controls.speed\r\n          * Math.min(timestampMs - lastSimTimestampMs, 16.67))\r\n          / 1000,\r\n      );\r\n      lastSimTimestampMs = timestampMs;\r\n    };\r\n\r\n    /**\r\n     * Paint the visualization\r\n     * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame\r\n     */\r\n    const paint = (timestampMs: number) => {\r\n      if (\r\n        this.simulation.controls.speed === 0\r\n        || this.simulation.controls.paused\r\n      ) {\r\n        requestAnimationFrame(paint);\r\n        renderer.clear();\r\n        renderer.render(this.scene!, camera);\r\n        viewHelper.render(renderer);\r\n        // labelRenderer.render(scene, camera);\r\n        orbitControls.update();\r\n        return;\r\n      }\r\n      step(timestampMs);\r\n\r\n      if (timePerFrame > 0 && timestampMs - lastPaint < timePerFrame) {\r\n        requestAnimationFrame(paint);\r\n        renderer.clear();\r\n        renderer.render(this.scene!, camera);\r\n        viewHelper.render(renderer);\r\n        // labelRenderer.render(scene, camera);\r\n        orbitControls.update();\r\n        return;\r\n      }\r\n\r\n      lastPaint = timestampMs;\r\n      if (this.simulation.showDebugInfo && stats) {\r\n        stats.update();\r\n      }\r\n\r\n      let ind = 0;\r\n      this.simulation.universes.forEach((u, i) => {\r\n        if (this.simulation.controls.showUniverse[u.label]) {\r\n          u.currState.bodies.forEach((b) => {\r\n            arr[ind].visible = true;\r\n            arr[ind].position.copy(b.position.clone()\r\n              .multiplyScalar(scale));\r\n            if (this.simulation.controls.showTrails) {\r\n              this.universeTrails[i].addTrail(arr[ind].position);\r\n            }\r\n            ind++;\r\n          });\r\n        } else {\r\n          u.currState.bodies.forEach((b) => {\r\n            arr[ind].visible = false;\r\n            ind++;\r\n          });\r\n        }\r\n      });\r\n      requestAnimationFrame(paint);\r\n      renderer.clear();\r\n      renderer.render(this.scene!, camera);\r\n      viewHelper.render(renderer);\r\n      // labelRenderer.render(scene, camera);\r\n      orbitControls.update();\r\n    };\r\n\r\n    requestAnimationFrame(paint);\r\n  }\r\n\r\n  /**\r\n   * Stop the simulation and visualization.\r\n   */\r\n  stop(): void {\r\n    this.scene?.clear();\r\n    this.scene = undefined;\r\n    this.universeTrails.forEach((ut) => {\r\n      ut.popAllTrails();\r\n    });\r\n    this.universeTrails = [];\r\n  }\r\n}\r\n\r\n/**\r\n * 2D recording visualizer using Plotly.\r\n * @category Visualizers\r\n */\r\nexport class RecordingVisualizer implements Visualizer {\r\n  simulation: Simulation;\r\n  divId: string = '';\r\n  universeTrails: PlotlyUniverseTrail[] = [];\r\n\r\n  /**\r\n   * Constructor for RealTimeVisualizer\r\n   * @param simulation simulation object\r\n   */\r\n  constructor(simulation: Simulation) {\r\n    this.simulation = simulation;\r\n  }\r\n\r\n  /**\r\n   * Adds default controls using lil-gui to the visualization.\r\n   * @param parentElement parent element to place the controller div in.\r\n   */\r\n  private addControls(parentElement: HTMLElement): void {\r\n    const gui = new GUI({\r\n      container: parentElement,\r\n    });\r\n    gui.domElement.style.position = 'absolute';\r\n    gui.domElement.style.top = '0';\r\n    gui.domElement.style.left = '0';\r\n    gui.domElement.style.zIndex = '1000';\r\n\r\n    const config = this.simulation.controls;\r\n    gui.add(config, 'speed');\r\n    gui.add(config, 'showTrails')\r\n      .onChange((value: boolean) => {\r\n        if (value === false) {\r\n          this.universeTrails.forEach((ut) => ut.popAllTrails());\r\n        }\r\n        config.showTrails = value;\r\n      });\r\n    const showUniverseFolder = gui.addFolder('Show Universe');\r\n    showUniverseFolder.open(false);\r\n    this.simulation.universes.forEach((u, i) => {\r\n      showUniverseFolder\r\n        .add(config.showUniverse, u.label)\r\n        .onChange((value: boolean) => {\r\n          if (value === false) {\r\n            this.universeTrails[i].popAllTrails();\r\n          }\r\n          config.showUniverse[u.label] = value;\r\n        });\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Simulate and play the visualization.\r\n   * @param divId div id to render the visualization in.\r\n   * @param width width of the visualization.\r\n   * @param height height of the visualization.\r\n   * @param recordFor number of seconds to record for..\r\n   */\r\n  start(divId: string, width: number, height: number, recordFor: number): void {\r\n    if (this.divId !== '') {\r\n      // throw new Error(\r\n      //   'Simulation already playing. Stop the current playtime before initiating a new one.',\r\n      // );\r\n      console.error('Simulation already playing. Stop the current playtime before initiating a new one.');\r\n      return;\r\n    }\r\n    this.divId = divId;\r\n    let element = document.getElementById(divId);\r\n    if (element === null) {\r\n      return;\r\n    }\r\n    // const width = element.clientWidth;\r\n    // const height = element.clientHeight;\r\n    let maxWidth = 0;\r\n    let maxHeight = 0;\r\n    this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => {\r\n      maxWidth = Math.max(maxWidth, Math.abs(b.position.x));\r\n      maxHeight = Math.max(maxHeight, Math.abs(b.position.y));\r\n    }));\r\n    const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth);\r\n\r\n    const recordedFrames: State[][] = [];\r\n    const totalFrames = this.simulation.maxFrameRate * recordFor;\r\n    let playInd = 1;\r\n    this.simulation.universes.forEach((u) => {\r\n      recordedFrames.push([u.currState.clone()]);\r\n    });\r\n    for (let i = 0; i < totalFrames; i++) {\r\n      this.simulation.simulateStep(1 / this.simulation.maxFrameRate);\r\n      this.simulation.universes.forEach((u, j) => {\r\n        recordedFrames[j].push(u.currState.clone());\r\n      });\r\n    }\r\n\r\n    const layout: Partial<Layout> = {\r\n      paper_bgcolor: '#000000',\r\n      plot_bgcolor: '#000000',\r\n      font: {\r\n        color: '#bfbfbf',\r\n      },\r\n      xaxis: {\r\n        autorange: false,\r\n        range: [-(width / 2) / scale, width / 2 / scale],\r\n      },\r\n      yaxis: {\r\n        autorange: false,\r\n        range: [-(height / 2) / scale, height / 2 / scale],\r\n      },\r\n      // uirevision: 'true',\r\n      showlegend: false,\r\n    };\r\n\r\n    if (this.simulation.controller === 'ui') {\r\n      this.addControls(element);\r\n    }\r\n\r\n    let stats: Stats | undefined;\r\n    if (this.simulation.showDebugInfo) {\r\n      stats = new Stats();\r\n      stats.dom.style.position = 'absolute';\r\n      stats.dom.style.bottom = '0px';\r\n      stats.dom.style.removeProperty('top');\r\n      element.appendChild(stats.dom);\r\n    }\r\n\r\n    const init_data: Data[] = this.simulation.universes.flatMap(\r\n      (uni: Universe): Data[] => {\r\n        const currTrail = new PlotlyUniverseTrail(\r\n          this.simulation.getMaxTrailLength(),\r\n          typeof uni.color === 'string' ? uni.color : uni.color[0],\r\n        );\r\n        this.universeTrails.push(currTrail);\r\n        const currData: Data = {\r\n          x: uni.currState.bodies.map((body) => body.position.x),\r\n          y: uni.currState.bodies.map((body) => body.position.y),\r\n          type: 'scatter',\r\n          mode: 'markers',\r\n          marker: {\r\n            color: uni.color,\r\n            sizemin: 6,\r\n            size: uni.currState.bodies.map((body) => Math.min(10, body.mass)),\r\n          },\r\n        };\r\n        if (this.simulation.getShowTrails()) {\r\n          uni.currState.bodies.forEach((b) => {\r\n            currTrail.addTrail(b.position.x, b.position.y);\r\n          });\r\n          return [currData, currTrail.data];\r\n        }\r\n        return [\r\n          currData,\r\n          {\r\n            x: [],\r\n            y: [],\r\n          },\r\n        ];\r\n      },\r\n    );\r\n\r\n    Plotly.newPlot(divId, init_data, layout, {\r\n      scrollZoom: true,\r\n      modeBarButtonsToRemove: [\r\n        'zoom2d',\r\n        'lasso2d',\r\n        'select2d',\r\n        'toImage',\r\n        'resetScale2d',\r\n      ],\r\n    });\r\n\r\n    if (animationId !== null) return;\r\n\r\n    /**\r\n     * Paint the visualization\r\n     * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame\r\n     */\r\n    const paint = (timestampMs: number) => {\r\n      if (\r\n        this.simulation.controls.speed === 0\r\n        || this.simulation.controls.paused\r\n      ) {\r\n        animationId = requestAnimationFrame(paint);\r\n        return;\r\n      }\r\n\r\n      const currPlayInd = Math.round(playInd);\r\n      const new_data = this.simulation.universes.flatMap(\r\n        (uni: Universe, i: number): Data[] => {\r\n          if (!this.simulation.getShowUniverse(uni.label)) {\r\n            return [\r\n              {\r\n                x: [],\r\n                y: [],\r\n              },\r\n              {},\r\n            ];\r\n          }\r\n          const currState = recordedFrames[i][currPlayInd];\r\n          const currData: Data = {\r\n            x: currState.bodies.map((body) => body.position.x),\r\n            y: currState.bodies.map((body) => body.position.y),\r\n            hovertext: currState.bodies.map((body) => body.label),\r\n            marker: {\r\n              size: currState.bodies.map((body) => Math.min(10, body.mass)),\r\n              color: uni.color,\r\n              sizemin: 6,\r\n            },\r\n            mode: 'markers',\r\n          };\r\n\r\n          let trailData: Data = {};\r\n          if (this.simulation.getShowTrails()) {\r\n            const currTrail = this.universeTrails[i];\r\n            currState.bodies.forEach((b): void => {\r\n              currTrail.addTrail(b.position.x, b.position.y);\r\n            });\r\n            trailData = currTrail.data;\r\n          }\r\n          return [currData, trailData];\r\n        },\r\n      );\r\n      Plotly.react(divId, new_data, layout);\r\n\r\n      if (this.simulation.showDebugInfo && stats) {\r\n        stats.update();\r\n      }\r\n\r\n      playInd = Math.round(playInd + this.simulation.controls.speed);\r\n      if (playInd < 0) {\r\n        if (this.simulation.looped) {\r\n          playInd = ((playInd % totalFrames) + totalFrames) % totalFrames;\r\n        } else {\r\n          playInd = 0;\r\n        }\r\n      } else if (playInd >= totalFrames) {\r\n        if (this.simulation.looped) {\r\n          playInd %= totalFrames;\r\n        } else {\r\n          playInd = totalFrames - 1;\r\n        }\r\n      }\r\n      animationId = requestAnimationFrame(paint);\r\n    };\r\n\r\n    animationId = requestAnimationFrame(paint);\r\n  }\r\n\r\n  /**\r\n   * Stop the simulation and visualization.\r\n   */\r\n  stop(): void {\r\n    Plotly.purge(this.divId);\r\n    this.divId = '';\r\n    this.universeTrails = [];\r\n  }\r\n}\r\n\r\n/**\r\n * 3D recording visualizer using Three.js.\r\n * @category Visualizers\r\n */\r\nexport class RecordingVisualizer3D implements Visualizer {\r\n  simulation: Simulation;\r\n  scene?: THREE.Scene;\r\n  universeTrails: ThreeUniverseTrail[] = [];\r\n\r\n  /**\r\n   * Constructor for RealTimeVisualizer3D.\r\n   * @param simulation simulation object.\r\n   */\r\n  constructor(simulation: Simulation) {\r\n    this.simulation = simulation;\r\n  }\r\n\r\n  /**\r\n   * Adds default controls to the visualization.\r\n   * @param parentElement parent element to place the controller div in.\r\n   */\r\n  private addControls(parentElement: HTMLElement) {\r\n    const gui = new GUI({\r\n      container: parentElement,\r\n    });\r\n    gui.domElement.style.position = 'absolute';\r\n    gui.domElement.style.top = '0';\r\n    gui.domElement.style.left = '0';\r\n    gui.domElement.style.zIndex = '1000';\r\n\r\n    const config = this.simulation.controls;\r\n    gui.add(config, 'speed');\r\n    gui.add(config, 'showTrails')\r\n      .onChange((value: boolean) => {\r\n        if (value === false) {\r\n          this.universeTrails.forEach((ut) => {\r\n            ut.popAllTrails();\r\n          });\r\n        }\r\n        config.showTrails = value;\r\n      });\r\n    const showUniverseFolder = gui.addFolder('Show Universe');\r\n    showUniverseFolder.open(false);\r\n    this.simulation.universes.forEach((u, i) => {\r\n      showUniverseFolder\r\n        .add(config.showUniverse, u.label)\r\n        .onChange((value: boolean) => {\r\n          if (value === false) {\r\n            this.universeTrails[i].popAllTrails();\r\n          }\r\n          config.showUniverse[u.label] = value;\r\n        });\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Simulate and play the visualization\r\n   * @param divId div id to render the visualization in.\r\n   * @param width width of the visualization.\r\n   * @param height height of the visualization.\r\n   * @param recordFor number of seconds to record for.\r\n   */\r\n  start(divId: string, width: number, height: number, recordFor: number): void {\r\n    if (this.scene !== undefined) {\r\n      // throw new Error(\r\n      //   'Simulation already playing. Stop the current playtime before initiating a new one.',\r\n      // );\r\n      console.error('Simulation already playing. Stop the current playtime before initiating a new one.');\r\n      return;\r\n    }\r\n    let element = document.getElementById(divId);\r\n    if (element === null) {\r\n      return;\r\n    }\r\n    // const width = element.clientWidth;\r\n    // const height = element.clientHeight;\r\n    let maxWidth = 0;\r\n    let maxHeight = 0;\r\n    this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => {\r\n      maxWidth = Math.max(maxWidth, Math.abs(b.position.x));\r\n      maxHeight = Math.max(maxHeight, Math.abs(b.position.y));\r\n    }));\r\n    const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth);\r\n\r\n    this.scene = new THREE.Scene();\r\n\r\n    const camera = new THREE.OrthographicCamera(\r\n      width / -2,\r\n      width / 2,\r\n      height / 2,\r\n      height / -2,\r\n      0,\r\n      10000000000,\r\n    );\r\n    camera.position.set(0, 0, Math.max(width, height));\r\n\r\n    const renderer = new THREE.WebGLRenderer();\r\n    renderer.setSize(width, height);\r\n    renderer.autoClear = false;\r\n    element.appendChild(renderer.domElement);\r\n\r\n    let stats: Stats | undefined;\r\n    if (this.simulation.showDebugInfo) {\r\n      stats = new Stats();\r\n      stats.dom.style.position = 'absolute';\r\n      stats.dom.style.right = '0px';\r\n      stats.dom.style.removeProperty('left');\r\n      element.appendChild(stats.dom);\r\n    }\r\n\r\n    if (this.simulation.controller === 'ui') {\r\n      this.addControls(element);\r\n    }\r\n\r\n    // const earthDiv = document.createElement('div');\r\n    // earthDiv.className = 'label';\r\n    // earthDiv.textContent = 'Earthhgkjfdghkjfgh';\r\n    // earthDiv.style.backgroundColor = 'transparent';\r\n    // earthDiv.style.color = 'white';\r\n    // earthDiv.style.fontFamily = 'sans-serif';\r\n    // earthDiv.style.background = 'rgba(0, 0, 0, 0.6)';\r\n\r\n    // const earthLabel = new CSS2DObject(earthDiv);\r\n    // earthLabel.position.set(0, 0, 0);\r\n    // earthLabel.center.set(0, 1);\r\n    // // arr[0].add(earthLabel);\r\n    // earthLabel.layers.set(0);\r\n    // const labelRenderer = new CSS2DRenderer();\r\n    // labelRenderer.setSize(width, height);\r\n    // labelRenderer.domElement.style.position = 'absolute';\r\n    // labelRenderer.domElement.style.top = '0px';\r\n    // element.appendChild(labelRenderer.domElement);\r\n\r\n    const orbitControls = new OrbitControls(camera, renderer.domElement);\r\n    orbitControls.listenToKeyEvents(window);\r\n    orbitControls.update();\r\n\r\n    const axesHelper = new THREE.AxesHelper(width);\r\n    this.scene.add(axesHelper);\r\n    const viewHelper = new ViewHelper(camera, renderer.domElement);\r\n\r\n    // var m: Map<string, THREE.LineSegments> = new Map();\r\n    let arr: THREE.LineSegments[] = [];\r\n\r\n    this.simulation.universes.forEach((u) => {\r\n      this.universeTrails.push(\r\n        new ThreeUniverseTrail(\r\n          this.simulation.maxTrailLength,\r\n          typeof u.color === 'string' ? u.color : u.color[0],\r\n          this.scene!,\r\n          scale,\r\n        ),\r\n      );\r\n      u.currState.bodies.forEach((b) => {\r\n        const sph = new THREE.SphereGeometry(\r\n          clipMinMax(Math.log2(b.mass) - 70, 10, 40),\r\n          8,\r\n          8,\r\n        );\r\n        const curr = new THREE.WireframeGeometry(sph);\r\n        const line = new THREE.LineSegments(\r\n          curr,\r\n          new THREE.LineBasicMaterial({\r\n            // @ts-ignore\r\n            color: new THREE.Color(u.color),\r\n          }),\r\n        );\r\n        this.scene!.add(line);\r\n        line.position.copy(b.position.clone()\r\n          .multiplyScalar(scale));\r\n        // m.set(u.label + \" \" + b.label, line);\r\n        arr.push(line);\r\n      });\r\n    });\r\n    // arr[0].add(earthLabel)\r\n\r\n    const recordedFrames: State[][] = [];\r\n    const totalFrames = this.simulation.maxFrameRate * recordFor;\r\n    let playInd = 1;\r\n    this.simulation.universes.forEach((u) => {\r\n      recordedFrames.push([u.currState.clone()]);\r\n    });\r\n    for (let i = 0; i < totalFrames; i++) {\r\n      this.simulation.simulateStep(1 / this.simulation.maxFrameRate);\r\n      this.simulation.universes.forEach((u, j) => {\r\n        recordedFrames[j].push(u.currState.clone());\r\n      });\r\n    }\r\n\r\n    /**\r\n     * Paint the visualization\r\n     * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame\r\n     */\r\n    const paint = (timestampMs: number) => {\r\n      if (\r\n        this.simulation.controls.speed === 0\r\n        || this.simulation.controls.paused\r\n      ) {\r\n        requestAnimationFrame(paint);\r\n        renderer.clear();\r\n        renderer.render(this.scene!, camera);\r\n        viewHelper.render(renderer);\r\n        // labelRenderer.render(scene, camera);\r\n        orbitControls.update();\r\n        return;\r\n      }\r\n\r\n      let ind = 0;\r\n      this.simulation.universes.forEach((u, i) => {\r\n        if (this.simulation.controls.showUniverse[u.label]) {\r\n          const currState = recordedFrames[i][playInd];\r\n          currState.bodies.forEach((b) => {\r\n            arr[ind].visible = true;\r\n            arr[ind].position.copy(b.position.clone()\r\n              .multiplyScalar(scale));\r\n            if (this.simulation.controls.showTrails) {\r\n              this.universeTrails[i].addTrail(arr[ind].position);\r\n            }\r\n            ind++;\r\n          });\r\n        } else {\r\n          u.currState.bodies.forEach(() => {\r\n            arr[ind].visible = false;\r\n            ind++;\r\n          });\r\n        }\r\n      });\r\n\r\n      if (this.simulation.showDebugInfo && stats) {\r\n        stats.update();\r\n      }\r\n\r\n      playInd = Math.round(playInd + this.simulation.controls.speed);\r\n      if (playInd < 0) {\r\n        if (this.simulation.looped) {\r\n          playInd = ((playInd % totalFrames) + totalFrames) % totalFrames;\r\n        } else {\r\n          playInd = 0;\r\n        }\r\n      } else if (playInd >= totalFrames) {\r\n        if (this.simulation.looped) {\r\n          playInd %= totalFrames;\r\n        } else {\r\n          playInd = totalFrames - 1;\r\n        }\r\n      }\r\n\r\n      requestAnimationFrame(paint);\r\n      renderer.clear();\r\n      renderer.render(this.scene!, camera);\r\n      viewHelper.render(renderer);\r\n      // labelRenderer.render(scene, camera);\r\n      orbitControls.update();\r\n    };\r\n\r\n    requestAnimationFrame(paint);\r\n  }\r\n\r\n  /**\r\n   * Stop the simulation and visualization.\r\n   */\r\n  stop(): void {\r\n    this.scene?.clear();\r\n    this.scene = undefined;\r\n    this.universeTrails.forEach((ut) => {\r\n      ut.popAllTrails();\r\n    });\r\n    this.universeTrails = [];\r\n  }\r\n}\r\n","/* eslint-disable jsdoc/require-param */\r\n/* eslint-disable jsdoc/check-param-names */\r\nimport { type Universe } from './Universe';\r\nimport { type Visualizer } from './Visualizer';\r\nimport {\r\n  RealTimeVisualizer,\r\n  RealTimeVisualizer3D,\r\n  RecordingVisualizer,\r\n  RecordingVisualizer3D,\r\n} from './library/Visualizer';\r\n\r\n/**\r\n * Visualization type.\r\n * @category Types\r\n */\r\nexport type VisType = '2D' | '3D';\r\n/**\r\n * Controller type.\r\n * - 'ui' for user interface control.\r\n * - 'code' for manual control via code.\r\n * - 'none' for no control.\r\n * @category Types\r\n */\r\nexport type ControllerType = 'ui' | 'code' | 'none';\r\n\r\n/**\r\n * A Simulation object that contains Universes and a Visualizer.\r\n * @category Building blocks\r\n */\r\nexport class Simulation {\r\n  /**\r\n   * Visualization object used to render the simulation.\r\n   * @hidden\r\n   */\r\n  readonly visualizer: Visualizer;\r\n  /**\r\n   * Array of Universes that make up this simulation. All universes are simulated independently and visualized together.\r\n   * @hidden\r\n   */\r\n  readonly universes: Universe[];\r\n  /**\r\n   * Controller type used to control the simulation.\r\n   * @hidden\r\n   */\r\n  readonly controller: ControllerType;\r\n  /**\r\n   * Maximum frame rate of the visualization.\r\n   * @hidden\r\n   */\r\n  maxFrameRate: number;\r\n  /**\r\n   * Maximum trail length used in the visualization.\r\n   * @hidden\r\n   */\r\n  maxTrailLength: number;\r\n  /**\r\n   * Whether the simulation is looped in case of a recording.\r\n   * @hidden\r\n   */\r\n  looped: boolean;\r\n\r\n  /**\r\n   * Controls object used to control the simulation.\r\n   * @hidden\r\n   */\r\n  controls: {\r\n    /**\r\n     * Speed of the simulation as a scale of normal time.\r\n     */\r\n    speed: number;\r\n    /**\r\n     * True if the simulation is paused.\r\n     */\r\n    paused: boolean;\r\n    /**\r\n     * True if trails are shown in the visualization.\r\n     */\r\n    showTrails: boolean;\r\n    /**\r\n     * Object containing whether each universe is shown in the visualization.\r\n     */\r\n    showUniverse: {\r\n      [key: string]: boolean;\r\n    };\r\n  } = {\r\n      speed: 1,\r\n      paused: true,\r\n      showTrails: false,\r\n      showUniverse: {},\r\n    };\r\n  /**\r\n   * True if debug info is shown in the visualization.\r\n   * @hidden\r\n   */\r\n  showDebugInfo: boolean;\r\n\r\n  /**\r\n   * Create a new Simulation object with the provided Universes and visualization config.\r\n   * @param universes array of Universes.\r\n   * @param visType visualization type.\r\n   * @param record whether to record the simulation.\r\n   * @param looped whether to loop the recorded simulation.\r\n   * @param controller controller type.\r\n   * @param showTrails whether to show trails in the visualization.\r\n   * @param showDebugInfo whether to show debug info in the visualization.\r\n   * @param maxFrameRate maximum frame rate of the visualization.\r\n   * @param maxTrailLength maximum trail for each universe.\r\n   */\r\n  constructor(\r\n    universes: Universe | Universe[],\r\n    {\r\n      visType = '2D',\r\n      record = false,\r\n      looped = true,\r\n      controller = 'none',\r\n      showTrails = false,\r\n      showDebugInfo = false,\r\n      maxFrameRate = -1,\r\n      maxTrailLength = 100,\r\n    }: {\r\n      visType?: VisType;\r\n      record?: boolean;\r\n      looped?: boolean;\r\n      controller?: ControllerType;\r\n      showTrails?: boolean;\r\n      showDebugInfo?: boolean;\r\n      maxFrameRate?: number;\r\n      maxTrailLength?: number;\r\n    },\r\n  ) {\r\n    this.universes = Array.isArray(universes) ? universes : [universes];\r\n    if (this.universes.length > 10) {\r\n      throw new Error('Too many universes');\r\n    }\r\n    const uniqueLabels = new Set(this.universes.map((u) => u.label));\r\n    if (uniqueLabels.size !== this.universes.length) {\r\n      throw new Error('Duplicate label in universes');\r\n    }\r\n    this.controller = controller;\r\n    this.universes.forEach((u) => {\r\n      this.controls.showUniverse[u.label] = true;\r\n    });\r\n    this.controls.showTrails = showTrails;\r\n    this.showDebugInfo = showDebugInfo;\r\n    this.maxFrameRate = maxFrameRate;\r\n    this.maxTrailLength = maxTrailLength;\r\n    this.looped = looped;\r\n    if (record) {\r\n      // if (this.universes.length > 1) {\r\n      //   throw new Error(\"Cannot record multiple universes\");\r\n      // }\r\n      this.maxFrameRate = 60;\r\n      this.visualizer\r\n        = visType === '2D'\r\n          ? new RecordingVisualizer(this)\r\n          : new RecordingVisualizer3D(this);\r\n    } else {\r\n      this.visualizer\r\n        = visType === '2D'\r\n          ? new RealTimeVisualizer(this)\r\n          : new RealTimeVisualizer3D(this);\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get the speed of the simulation.\r\n   * @returns speed of the simulation as a scale of normal time.\r\n   */\r\n  getSpeed(): number {\r\n    return this.controls.speed;\r\n  }\r\n\r\n  /**\r\n   * Set the speed of the simulation. Only works if the controller is 'code'.\r\n   * @param speed speed of the simulation as a scale of normal time.\r\n   */\r\n  setSpeed(speed: number) {\r\n    if (this.controller === 'code') {\r\n      this.controls.speed = speed;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get whether the simulation is playing.\r\n   * @returns true if the simulation is playing.\r\n   */\r\n  isPlaying(): boolean {\r\n    return !this.controls.paused;\r\n  }\r\n\r\n  /**\r\n   * Pause the simulation. Only works if the controller is 'code'.\r\n   */\r\n  pause(): void {\r\n    if (this.controller === 'code') {\r\n      this.controls.paused = true;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Resume the simulation. Only works if the controller is 'code'.\r\n   */\r\n  resume(): void {\r\n    if (this.controller === 'code') {\r\n      this.controls.paused = false;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get whether trails are shown in the visualization.\r\n   * @returns true if trails are shown.\r\n   */\r\n  getShowTrails(): boolean {\r\n    return this.controls.showTrails;\r\n  }\r\n\r\n  /**\r\n   * Set whether to show trails in the visualization. Only works if the controller is 'code'.\r\n   * @param showTrails true to show trails.\r\n   */\r\n  setShowTrails(showTrails: boolean): void {\r\n    if (this.controller === 'code') {\r\n      this.controls.showTrails = showTrails;\r\n      if (!showTrails) {\r\n        // TODO\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * True if the universe with the given label is shown.\r\n   * @param label universe label.\r\n   * @returns whether the universe is shown.\r\n   */\r\n  getShowUniverse(label: string): boolean {\r\n    return this.controls.showUniverse[label];\r\n  }\r\n\r\n  /**\r\n   * Set whether to show the universe with the given label. Only works if the controller is 'code'.\r\n   * @param label universe label.\r\n   * @param show true to show the universe.\r\n   */\r\n  setShowUniverse(label: string, show: boolean) {\r\n    if (this.controller === 'code') {\r\n      this.controls.showUniverse[label] = show;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Get the maximum trail length used in the visualization.\r\n   * @returns maximum trail length.\r\n   */\r\n  getMaxTrailLength(): number {\r\n    return this.maxTrailLength;\r\n  }\r\n\r\n  /**\r\n   * Set the maximum trail length used in the visualization. Changes only apply on the next Simulation.play() call.\r\n   * @param maxTrailLength maximum trail length.\r\n   */\r\n  setMaxTrailLength(maxTrailLength: number): void {\r\n    if (this.controller === 'code') {\r\n      this.maxTrailLength = maxTrailLength;\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Simulates a single step in this simulation.\r\n   * @param deltaT time step to simulate.\r\n   * @hidden\r\n   */\r\n  simulateStep(deltaT: number): void {\r\n    this.universes.forEach((universe) => {\r\n      universe.simulateStep(deltaT);\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Insert the simulation visualization in the div with the given id.\r\n   * @param divId div id.\r\n   * @param speed initial time scale.\r\n   * @param paused whether to start the simulation paused.\r\n   * @param recordFor number of seconds to record for, only used if in record mode.\r\n   */\r\n  start(\r\n    divId: string,\r\n    width: number,\r\n    height: number,\r\n    speed: number = 1,\r\n    paused: boolean = false,\r\n    recordFor: number = 0,\r\n  ): void {\r\n    if (recordFor === undefined) {\r\n      throw new Error('recordFor must be defined if record is true');\r\n    }\r\n    this.controls.paused = paused;\r\n    this.controls.speed = speed;\r\n    this.visualizer.start(divId, width, height, recordFor);\r\n  }\r\n\r\n  /**\r\n   * Stop and clear the simulation.\r\n   */\r\n  stop(): void {\r\n    this.visualizer.stop();\r\n  }\r\n}\r\n","import { Vector3 } from 'three';\r\nimport { type State } from '../State';\r\nimport { type Transformation } from '../Transformation';\r\n\r\n/**\r\n * Frame of reference transformation to the center of the first body in the system.\r\n * @category Transformations\r\n */\r\nexport class BodyCenterTransformation implements Transformation {\r\n  /**\r\n   * Transform the frame of reference to the center of the first body in the system.\r\n   * @param state state to transform.\r\n   * @returns transformed state.\r\n   */\r\n  transform(state: State): State {\r\n    const transform = state.bodies[0].position.clone();\r\n    state.bodies.forEach((b) => {\r\n      b.position.sub(transform);\r\n    });\r\n    return state;\r\n  }\r\n}\r\n\r\n/**\r\n * Frame of reference transformation to the center of mass of the system.\r\n * @category Transformations\r\n */\r\nexport class CoMTransformation implements Transformation {\r\n  /**\r\n   * Transform the frame of reference to the center of mass of the system.\r\n   * @param state state to transform.\r\n   * @returns transformed state.\r\n   */\r\n  transform(state: State): State {\r\n    let totalMass = 0;\r\n    let com = new Vector3();\r\n    state.bodies.forEach((b) => {\r\n      totalMass += b.mass;\r\n      com.add(b.position.clone()\r\n        .multiplyScalar(b.mass));\r\n    });\r\n    com.divideScalar(totalMass);\r\n    state.bodies.forEach((b) => {\r\n      b.position.sub(com);\r\n    });\r\n    return state;\r\n  }\r\n}\r\n\r\n/**\r\n * Frame of reference transformation around an axis by an angle. Makes sense to this transformation only during initialization of the universe and not at every time step.\r\n * @category Transformations\r\n */\r\nexport class RotateTransformation implements Transformation {\r\n  readonly axis: Vector3;\r\n  readonly angle: number;\r\n\r\n  /**\r\n   * Create a new RotateTransformer.\r\n   * @param axis axis to rotate around.\r\n   * @param angle angle to rotate by.\r\n   */\r\n  constructor(axis: Vector3, angle: number) {\r\n    this.axis = axis;\r\n    this.angle = angle;\r\n  }\r\n\r\n  /**\r\n   * Transform the frame of reference around an axis by an angle.\r\n   * @param state state to transform.\r\n   * @returns transformed state.\r\n   */\r\n  transform(state: State): State {\r\n    state.bodies.forEach((b) => {\r\n      b.position.applyAxisAngle(this.axis, this.angle);\r\n      b.velocity.applyAxisAngle(this.axis, this.angle);\r\n      b.acceleration.applyAxisAngle(this.axis, this.angle);\r\n    });\r\n    return state;\r\n  }\r\n}\r\n\r\n// export class PinTransformer implements Transformer {\r\n//   readonly axis: Vector3;\r\n//   readonly index: number;\r\n\r\n//   constructor(axis: Vector3, index: number) {\r\n//     this.axis = axis;\r\n//     this.index = index\r\n//   }\r\n\r\n//   transform(state: State): State {\r\n//     const angle = state.bodies[this.index].position.clone().angleTo(this.axis);\r\n//     const pivot = state.bodies[this.index].position.clone().cross(this.axis.clone()).normalize();\r\n//     state.bodies.forEach((b) => {\r\n//       b.position.applyAxisAngle(pivot.clone(), angle);\r\n//       b.velocity.applyAxisAngle(pivot.clone(), angle);\r\n//       b.acceleration.applyAxisAngle(pivot.clone(), angle);\r\n//     });\r\n//     return state;\r\n//   }\r\n// }\r\n","import { type State } from './State';\r\n\r\n/**\r\n * Represents a Frame of Reference transformation.\r\n * @category Interfaces\r\n * @category Transformations\r\n */\r\nexport interface Transformation {\r\n  /**\r\n   * Transform the state to a new frame of reference.\r\n   * @param state state to transform.\r\n   * @param deltaT time step taken to get to this state. Only applicable for time-dependent transformations.\r\n   * @returns transformed state.\r\n   */\r\n  transform(state: State, deltaT: number): State;\r\n}\r\n\r\n/**\r\n * A Frame of Reference transformation that uses the user-defined lambda function.\r\n * @category Transformations\r\n */\r\nexport class LambdaTransformation implements Transformation {\r\n  readonly fn: (state: State, deltaT: number) => State;\r\n\r\n  /**\r\n   * Create a new LambdaTransformer with the provided lambda function.\r\n   * - Lambda function should take in the current state and the time step and return the transformed state.\r\n   * - The time step is only applicable for time-dependent transformations.\r\n   * - Transformed state should contain the same number of bodies as the input state, and the order should be preserved.\r\n   * @param fn lambda function.\r\n   */\r\n  constructor(fn: (state: State, deltaT: number) => State) {\r\n    this.fn = fn;\r\n  }\r\n\r\n  /**\r\n   * Transform the state's frame of reference using the lambda function.\r\n   * @param state state to transform.\r\n   * @param deltaT time step taken to get to this state. Only applicable for time-dependent transformations.\r\n   * @returns transformed state.\r\n   */\r\n  transform(state: State, deltaT: number): State {\r\n    return this.fn(state, deltaT);\r\n  }\r\n}\r\n","import { type SimulateFunction } from './SimulateFunction';\r\nimport { type State } from './State';\r\nimport { type Transformation } from './Transformation';\r\n\r\n/**\r\n * Container for a Universe's configuration.\r\n * @category Types\r\n */\r\nexport type UniverseConfig = {\r\n  /**\r\n   * Previous state of the Universe.\r\n   */\r\n  prevState: State;\r\n  /**\r\n   * Current state of the Universe.\r\n   */\r\n  currState: State;\r\n  /**\r\n   * Color of the bodies in the Universe. A single color applied to all bodies or an array of colors applied to each body respectively. Length of the array should match the number of bodies in the state.\r\n   */\r\n  color: string | string[];\r\n  /**\r\n   * Label of the Universe.\r\n   */\r\n  label: string;\r\n  /**\r\n   * Simulation function used to simulate the Universe.\r\n   */\r\n  simFunc: SimulateFunction;\r\n  /**\r\n   * Array of transformations to be applied to the Universe's state after simulation and before visualization.\r\n   */\r\n  transformations: Transformation[];\r\n};\r\n\r\n/**\r\n * A Universe object that contains previous and current state of the universe, a simulation function, frame of reference transformations and other necessary data.\r\n * @category Building blocks\r\n */\r\nexport class Universe {\r\n  prevState: State;\r\n  currState: State;\r\n  /**\r\n   * Color of the bodies in the Universe. A single color applied to all bodies or an array of colors applied to each body respectively. Incase of array, length should match the number of bodies in the state.\r\n   */\r\n  color: string | string[];\r\n  /**\r\n   * Label of the Universe.\r\n   */\r\n  label: string;\r\n  /**\r\n   * Simulation function used to simulate the Universe.\r\n   */\r\n  simFunc: SimulateFunction;\r\n  /**\r\n   * Array of transformations to be applied to the Universe's state after simulation and before visualization.\r\n   */\r\n  transformations: Transformation[];\r\n\r\n  /**\r\n   * Create a new Universe with the provided configuration.\r\n   * @param config configuration object.\r\n   */\r\n  constructor(config: Partial<UniverseConfig>) {\r\n    if (config.currState === undefined) throw new Error('Missing Current State in Universe');\r\n    if (config.simFunc === undefined) throw new Error('Missing Simulation Function in Universe');\r\n    this.label = config.label === undefined ? 'Universe' : config.label;\r\n    this.prevState\r\n      = config.prevState === undefined ? config.currState : config.prevState;\r\n    this.currState = config.currState;\r\n    this.color\r\n      = config.color === undefined ? 'rgba(255, 255, 255, 1)' : config.color;\r\n    this.simFunc = config.simFunc;\r\n    this.transformations\r\n      = config.transformations === undefined\r\n        ? []\r\n        : Array.isArray(config.transformations)\r\n          ? config.transformations\r\n          : [config.transformations];\r\n  }\r\n\r\n  /**\r\n   * Simulate a step in the Universe using the SimulateFunction and Transformations.\r\n   * @param deltaT time step to simulate.\r\n   * @hidden\r\n   */\r\n  simulateStep(deltaT: number) {\r\n    let newState = this.simFunc.simulate(\r\n      deltaT,\r\n      this.currState,\r\n      this.prevState,\r\n    );\r\n    this.prevState = this.currState;\r\n    this.transformations.forEach((t) => {\r\n      newState = t.transform(newState, deltaT);\r\n    });\r\n    this.currState = newState;\r\n  }\r\n\r\n  /**\r\n   * Deep copy the current Universe.\r\n   * @returns a new Universe instance.\r\n   */\r\n  clone(): Universe {\r\n    return new Universe({\r\n      prevState: this.prevState.clone(),\r\n      currState: this.currState.clone(),\r\n      color: this.color,\r\n      label: this.label,\r\n      simFunc: this.simFunc,\r\n      transformations: this.transformations,\r\n    });\r\n  }\r\n}\r\n"],"names":["root","factory","exports","module","require","define","amd","this","__WEBPACK_EXTERNAL_MODULE__604__","__WEBPACK_EXTERNAL_MODULE__924__","__WEBPACK_EXTERNAL_MODULE__987__","__WEBPACK_EXTERNAL_MODULE__686__","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","undefined","__webpack_modules__","n","getter","__esModule","d","a","definition","key","o","Object","defineProperty","enumerable","get","obj","prop","prototype","hasOwnProperty","call","r","Symbol","toStringTag","value","CelestialBody","constructor","label","mass","position","velocity","acceleration","clone","Gravity","G","getForces","bodies","length","ans","i","push","Vector3","j","currForce","calcNewtonian","add","sub","b","distSq","distanceToSquared","forceVal","normalize","multiplyScalar","CentripetalForce","center","map","body","directionVector","setLength","lengthSq","CombinedForce","forces","forEach","force","val","index","State","VelocityVerletSim","forceCalculator","simulate","deltaT","currState","updatedBodies","v","updatedBody","calcNewPos","newForces","newAccel","divideScalar","currPos","currVel","currAccel","ExplicitEulerSim","rateUpdate","updatedForces","prev","rate","SemiImplicitEulerSim","updatedVel","RungeKutta4Sim","weights","Error","bodiesK","kv","kx","k2v","getInterKV","k2x","getInterKX","k3v","k3x","k4v","k4x","weightedXRate","weightedVRate","krInd","newBodies","newBody","kInd","LambdaSim","fn","prevState","Controller","parent","object","property","className","elementType","_disabled","_hidden","initialValue","getValue","domElement","document","createElement","classList","$name","nextNameID","id","$widget","$disable","appendChild","addEventListener","e","stopPropagation","children","controllers","$children","_listenCallback","bind","name","_name","textContent","onChange","callback","_onChange","_callOnChange","_changed","onFinishChange","_onFinishChange","_callOnFinishChange","reset","setValue","enable","enabled","disable","disabled","toggle","toggleAttribute","show","style","display","hide","options","controller","destroy","min","max","step","decimals","listen","_listening","_listenCallbackID","cancelAnimationFrame","requestAnimationFrame","curValue","save","_listenPrevValue","updateDisplay","load","splice","indexOf","removeChild","BooleanController","super","$input","setAttribute","checked","normalizeColorString","string","match","result","parseInt","toString","padStart","INT","isPrimitive","fromHexString","substring","toHexString","ARRAY","Array","isArray","target","rgbScale","int","g","OBJECT","FORMATS","ColorController","$text","$display","_format","find","format","_rgbScale","_initialValueHexString","_textFocused","_setValueFromHexString","tryParse","select","newValue","backgroundColor","FunctionController","$button","preventDefault","passive","NumberController","_initInput","stepExplicit","_getImplicitStep","_decimals","_min","_onUpdateMinMax","_max","explicit","_step","_stepExplicit","_hasSlider","percent","Math","$fill","width","_inputFocused","toFixed","window","matchMedia","matches","increment","delta","parseFloat","isNaN","_snapClampSetValue","initClientX","initClientY","prevClientY","initValue","dragDelta","testingForVerticalDrag","onMouseMove","dx","clientX","dy","clientY","abs","blur","_setDraggingStyle","onMouseUp","_arrowKeyMultiplier","removeEventListener","_snap","_clamp","code","_normalizeMouseWheel","_initSlider","$slider","insertBefore","setValueFromX","rect","getBoundingClientRect","left","right","c","mouseMove","mouseUp","prevClientX","testingForScroll","beginTouchDrag","touches","onTouchMove","onTouchEnd","callOnFinishChange","wheelFinishChangeTimeout","_hasScrollBar","deltaX","deltaY","clearTimeout","setTimeout","active","axis","_hasMin","_hasMax","floor","wheelDelta","mult","shiftKey","altKey","round","toPrecision","scrollHeight","clientHeight","OptionController","$select","_values","selectedIndex","remove","values","_names","keys","replaceChildren","$option","StringController","stylesInjected","GUI","autoPlace","container","title","closeFolders","injectStyles","touchStyles","folders","_closed","$title","openAnimated","click","cssContent","injected","innerHTML","before","querySelector","head","_injectStyles","setProperty","_closeFolders","$1","console","error","addColor","addFolder","folder","close","recursive","f","_title","open","_setClosed","closed","_callOnOpenClose","initialHeight","height","onTransitionEnd","targetHeight","controllersRecursive","onOpenClose","_onOpenClose","changedGUI","parentElement","from","concat","foldersRecursive","animationId","clipMinMax","x","PlotlyUniverseTrail","maxTrailLength","color","data","y","mode","marker","size","trailLength","trailInd","addTrail","popAllTrails","RealTimeVisualizer","simulation","divId","universeTrails","addControls","gui","top","zIndex","config","controls","ut","showTrails","showUniverseFolder","universes","u","showUniverse","start","element","getElementById","maxWidth","maxHeight","scale","layout","paper_bgcolor","plot_bgcolor","font","xaxis","autorange","range","yaxis","showlegend","stats","showDebugInfo","dom","bottom","removeProperty","init_data","flatMap","uni","currTrail","getMaxTrailLength","currData","type","sizemin","getShowTrails","scrollZoom","modeBarButtonsToRemove","timePerFrame","maxFrameRate","lastPaintTimestampMs","lastSimTimestampMs","timestampMs","simulateStep","speed","paint","paused","new_data","getShowUniverse","hovertext","trailData","update","stop","ThreeUniverseTrail","scene","geometry","Float32Array","trails","pos","visible","posArray","set","attributes","array","toArray","needsUpdate","RealTimeVisualizer3D","camera","renderer","setSize","autoClear","orbitControls","OrbitControls","listenToKeyEvents","axesHelper","viewHelper","ViewHelper","arr","sph","log2","curr","line","copy","performance","now","lastPaint","clear","render","ind","RecordingVisualizer","recordFor","recordedFrames","totalFrames","playInd","currPlayInd","looped","RecordingVisualizer3D","Simulation","visType","record","Set","visualizer","getSpeed","setSpeed","isPlaying","pause","resume","setShowTrails","setShowUniverse","setMaxTrailLength","universe","BodyCenterTransformation","transform","state","CoMTransformation","totalMass","com","RotateTransformation","angle","applyAxisAngle","LambdaTransformation","Universe","simFunc","transformations","newState","t"],"sourceRoot":""} \ No newline at end of file diff --git a/dist/index.js.LICENSE.txt b/dist/index.js.LICENSE.txt deleted file mode 100644 index 57bb07f..0000000 --- a/dist/index.js.LICENSE.txt +++ /dev/null @@ -1,7 +0,0 @@ -/** - * lil-gui - * https://lil-gui.georgealways.com - * @version 0.19.2 - * @author George Michael Brower - * @license MIT - */ diff --git a/dist/src/CelestialBody.js b/dist/src/CelestialBody.js new file mode 100644 index 0000000..c322895 --- /dev/null +++ b/dist/src/CelestialBody.js @@ -0,0 +1,32 @@ +/** + * Represents a celestial body with all of its kinematic properties. + * @category Building blocks + */ +export class CelestialBody { + /** + * Create a new CelestialBody with the provided information. + * @param label label of the body. + * @param mass mass of the body. + * @param position position of the body. + * @param velocity velocity of the body. + * @param acceleration acceleration of the body. + */ + constructor(label, mass, position, velocity, acceleration) { + this.label = label; + this.mass = mass; + this.position = position; + this.velocity = velocity; + this.acceleration = acceleration; + } + /** + * Deep copy the current CelestialBody with the updated kinematic properties. + * @param position new position. + * @param velocity new velocity. + * @param acceleration new acceleration. + * @returns a new CelestialBody instance with the updated properties. + */ + clone(position, velocity, acceleration) { + return new CelestialBody(this.label, this.mass, position === undefined ? this.position.clone() : position, velocity === undefined ? this.velocity.clone() : velocity, acceleration === undefined ? this.acceleration.clone() : acceleration); + } +} +//# sourceMappingURL=CelestialBody.js.map \ No newline at end of file diff --git a/dist/src/CelestialBody.js.map b/dist/src/CelestialBody.js.map new file mode 100644 index 0000000..e57d037 --- /dev/null +++ b/dist/src/CelestialBody.js.map @@ -0,0 +1 @@ +{"version":3,"file":"CelestialBody.js","sourceRoot":"","sources":["../../src/CelestialBody.ts"],"names":[],"mappings":"AAEA;;;GAGG;AACH,MAAM,OAAO,aAAa;IAsBxB;;;;;;;OAOG;IACH,YACE,KAAa,EACb,IAAY,EACZ,QAAiB,EACjB,QAAiB,EACjB,YAAqB;QAErB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;QACnB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;IACnC,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,QAAkB,EACtB,QAAkB,EAClB,YAAsB;QACtB,OAAO,IAAI,aAAa,CACtB,IAAI,CAAC,KAAK,EACV,IAAI,CAAC,IAAI,EACT,QAAQ,KAAK,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,QAAQ,EACzD,QAAQ,KAAK,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,QAAQ,EACzD,YAAY,KAAK,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,YAAY,CACtE,CAAC;IACJ,CAAC;CACF"} \ No newline at end of file diff --git a/dist/src/Force.js b/dist/src/Force.js new file mode 100644 index 0000000..69d14e3 --- /dev/null +++ b/dist/src/Force.js @@ -0,0 +1,25 @@ +/** + * Function object that uses the user-defined lambda function to calculate the forces acting on the bodies. + * @category Forces + */ +export class LambdaForce { + /** + * Create a new LambdaForce with the provided lambda function. + * - Lambda function should take in an array of CelestialBodies and return an array of forces acting on the bodies. + * - arr[i] should represent the force acting on the ith body. + * - Length of the returned array should be equal to the length of the input array of CelestialBodies. + * @param fn lambda function. + */ + constructor(fn) { + this.fn = fn; + } + /** + * Get the forces acting on the bodies. + * @param bodies array of CelestialBodies. + * @returns array of forces acting on the bodies. + */ + getForces(bodies) { + return this.fn(bodies); + } +} +//# sourceMappingURL=Force.js.map \ No newline at end of file diff --git a/dist/src/Force.js.map b/dist/src/Force.js.map new file mode 100644 index 0000000..c245114 --- /dev/null +++ b/dist/src/Force.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Force.js","sourceRoot":"","sources":["../../src/Force.ts"],"names":[],"mappings":"AAYA;;;GAGG;AACH,MAAM,OAAO,WAAW;IAMtB;;;;;;OAMG;IACH,YAAY,EAA0C;QACpD,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;IACf,CAAC;IAGD;;;;OAIG;IACH,SAAS,CAAC,MAAuB;QAC/B,OAAO,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;IACzB,CAAC;CACF"} \ No newline at end of file diff --git a/dist/src/SimulateFunction.js b/dist/src/SimulateFunction.js new file mode 100644 index 0000000..fc94acf --- /dev/null +++ b/dist/src/SimulateFunction.js @@ -0,0 +1,26 @@ +/** + * Function object that uses the user-defined lambda function to simulate the Universe. + * @category SimulateFunctions + */ +export class LambdaSim { + /** + * Create a new LambdaSim with the provided lambda function. + * - The lambda function should take in the time step, the current state of the Universe, and the previous state of the Universe, and return the next state of the Universe. + * - The lambda function should call or calculate the forces action on the bodies by itself. + * @param fn lambda function. + */ + constructor(fn) { + this.fn = fn; + } + /** + * Simulate the Universe using the lambda function. + * @param deltaT time step. + * @param currState current state of the Universe. + * @param prevState previous state of the Universe. + * @returns the next state of the Universe. + */ + simulate(deltaT, currState, prevState) { + return this.fn(deltaT, currState, prevState); + } +} +//# sourceMappingURL=SimulateFunction.js.map \ No newline at end of file diff --git a/dist/src/SimulateFunction.js.map b/dist/src/SimulateFunction.js.map new file mode 100644 index 0000000..3deeede --- /dev/null +++ b/dist/src/SimulateFunction.js.map @@ -0,0 +1 @@ +{"version":3,"file":"SimulateFunction.js","sourceRoot":"","sources":["../../src/SimulateFunction.ts"],"names":[],"mappings":"AAmBA;;;GAGG;AACH,MAAM,OAAO,SAAS;IAGpB;;;;;OAKG;IACH,YAAY,EAAiE;QAC3E,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;IACf,CAAC;IAED;;;;;;OAMG;IACH,QAAQ,CAAC,MAAc,EAAE,SAAgB,EAAE,SAAgB;QACzD,OAAO,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC;IAC/C,CAAC;CACF"} \ No newline at end of file diff --git a/dist/src/Simulation.js b/dist/src/Simulation.js new file mode 100644 index 0000000..0444a70 --- /dev/null +++ b/dist/src/Simulation.js @@ -0,0 +1,188 @@ +import { RealTimeVisualizer, RealTimeVisualizer3D, RecordingVisualizer, RecordingVisualizer3D, } from './library/Visualizer'; +/** + * A Simulation object that contains Universes and a Visualizer. + * @category Building blocks + */ +export class Simulation { + /** + * Create a new Simulation object with the provided Universes and visualization config. + * @param universes array of Universes. + * @param visType visualization type. + * @param record whether to record the simulation. + * @param looped whether to loop the recorded simulation. + * @param controller controller type. + * @param showTrails whether to show trails in the visualization. + * @param showDebugInfo whether to show debug info in the visualization. + * @param maxFrameRate maximum frame rate of the visualization. + * @param maxTrailLength maximum trail for each universe. + */ + constructor(universes, { visType = '2D', record = false, looped = true, controller = 'none', showTrails = false, showDebugInfo = false, maxFrameRate = -1, maxTrailLength = 100, }) { + /** + * Controls object used to control the simulation. + * @hidden + */ + this.controls = { + speed: 1, + paused: true, + showTrails: false, + showUniverse: {}, + }; + this.universes = Array.isArray(universes) ? universes : [universes]; + if (this.universes.length > 10) { + throw new Error('Too many universes'); + } + const uniqueLabels = new Set(this.universes.map((u) => u.label)); + if (uniqueLabels.size !== this.universes.length) { + throw new Error('Duplicate label in universes'); + } + this.controller = controller; + this.universes.forEach((u) => { + this.controls.showUniverse[u.label] = true; + }); + this.controls.showTrails = showTrails; + this.showDebugInfo = showDebugInfo; + this.maxFrameRate = maxFrameRate; + this.maxTrailLength = maxTrailLength; + this.looped = looped; + if (record) { + // if (this.universes.length > 1) { + // throw new Error("Cannot record multiple universes"); + // } + this.maxFrameRate = 60; + this.visualizer + = visType === '2D' + ? new RecordingVisualizer(this) + : new RecordingVisualizer3D(this); + } + else { + this.visualizer + = visType === '2D' + ? new RealTimeVisualizer(this) + : new RealTimeVisualizer3D(this); + } + } + /** + * Get the speed of the simulation. + * @returns speed of the simulation as a scale of normal time. + */ + getSpeed() { + return this.controls.speed; + } + /** + * Set the speed of the simulation. Only works if the controller is 'code'. + * @param speed speed of the simulation as a scale of normal time. + */ + setSpeed(speed) { + if (this.controller === 'code') { + this.controls.speed = speed; + } + } + /** + * Get whether the simulation is playing. + * @returns true if the simulation is playing. + */ + isPlaying() { + return !this.controls.paused; + } + /** + * Pause the simulation. Only works if the controller is 'code'. + */ + pause() { + if (this.controller === 'code') { + this.controls.paused = true; + } + } + /** + * Resume the simulation. Only works if the controller is 'code'. + */ + resume() { + if (this.controller === 'code') { + this.controls.paused = false; + } + } + /** + * Get whether trails are shown in the visualization. + * @returns true if trails are shown. + */ + getShowTrails() { + return this.controls.showTrails; + } + /** + * Set whether to show trails in the visualization. Only works if the controller is 'code'. + * @param showTrails true to show trails. + */ + setShowTrails(showTrails) { + if (this.controller === 'code') { + this.controls.showTrails = showTrails; + if (!showTrails) { + // TODO + } + } + } + /** + * True if the universe with the given label is shown. + * @param label universe label. + * @returns whether the universe is shown. + */ + getShowUniverse(label) { + return this.controls.showUniverse[label]; + } + /** + * Set whether to show the universe with the given label. Only works if the controller is 'code'. + * @param label universe label. + * @param show true to show the universe. + */ + setShowUniverse(label, show) { + if (this.controller === 'code') { + this.controls.showUniverse[label] = show; + } + } + /** + * Get the maximum trail length used in the visualization. + * @returns maximum trail length. + */ + getMaxTrailLength() { + return this.maxTrailLength; + } + /** + * Set the maximum trail length used in the visualization. Changes only apply on the next Simulation.play() call. + * @param maxTrailLength maximum trail length. + */ + setMaxTrailLength(maxTrailLength) { + if (this.controller === 'code') { + this.maxTrailLength = maxTrailLength; + } + } + /** + * Simulates a single step in this simulation. + * @param deltaT time step to simulate. + * @hidden + */ + simulateStep(deltaT) { + this.universes.forEach((universe) => { + universe.simulateStep(deltaT); + }); + } + /** + * Insert the simulation visualization in the div with the given id. + * @param divId div id. + * @param speed initial time scale. + * @param paused whether to start the simulation paused. + * @param recordFor number of seconds to record for, only used if in record mode. + */ + start(divId, width, height, speed = 1, paused = false, recordFor = 0) { + if (recordFor === undefined) { + throw new Error('recordFor must be defined if record is true'); + } + this.controls.paused = paused; + this.controls.speed = speed; + this.visualizer.start(divId, width, height, recordFor); + } + /** + * Stop and clear the simulation. + */ + stop() { + this.visualizer.stop(); + } +} +//# sourceMappingURL=Simulation.js.map \ No newline at end of file diff --git a/dist/src/Simulation.js.map b/dist/src/Simulation.js.map new file mode 100644 index 0000000..5302558 --- /dev/null +++ b/dist/src/Simulation.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Simulation.js","sourceRoot":"","sources":["../../src/Simulation.ts"],"names":[],"mappings":"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"} \ No newline at end of file diff --git a/dist/src/State.js b/dist/src/State.js new file mode 100644 index 0000000..20aa829 --- /dev/null +++ b/dist/src/State.js @@ -0,0 +1,21 @@ +/** + * Represents a Universe's state snapshot. + * @category Building blocks + */ +export class State { + /** + * Create a new State with the given celestial bodies. + * @param bodies array of celestial bodies. + */ + constructor(bodies) { + this.bodies = bodies; + } + /** + * Deep copy this state + * @returns a new State instance. + */ + clone() { + return new State(this.bodies.map((body) => body.clone())); + } +} +//# sourceMappingURL=State.js.map \ No newline at end of file diff --git a/dist/src/State.js.map b/dist/src/State.js.map new file mode 100644 index 0000000..7bd3eba --- /dev/null +++ b/dist/src/State.js.map @@ -0,0 +1 @@ +{"version":3,"file":"State.js","sourceRoot":"","sources":["../../src/State.ts"],"names":[],"mappings":"AAEA;;;GAGG;AACH,MAAM,OAAO,KAAK;IAMhB;;;OAGG;IACH,YAAY,MAAuB;QACjC,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;IAED;;;OAGG;IACH,KAAK;QACH,OAAO,IAAI,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;IAC5D,CAAC;CACF"} \ No newline at end of file diff --git a/dist/src/Transformation.js b/dist/src/Transformation.js new file mode 100644 index 0000000..30a7143 --- /dev/null +++ b/dist/src/Transformation.js @@ -0,0 +1,26 @@ +/** + * A Frame of Reference transformation that uses the user-defined lambda function. + * @category Transformations + */ +export class LambdaTransformation { + /** + * Create a new LambdaTransformer with the provided lambda function. + * - Lambda function should take in the current state and the time step and return the transformed state. + * - The time step is only applicable for time-dependent transformations. + * - Transformed state should contain the same number of bodies as the input state, and the order should be preserved. + * @param fn lambda function. + */ + constructor(fn) { + this.fn = fn; + } + /** + * Transform the state's frame of reference using the lambda function. + * @param state state to transform. + * @param deltaT time step taken to get to this state. Only applicable for time-dependent transformations. + * @returns transformed state. + */ + transform(state, deltaT) { + return this.fn(state, deltaT); + } +} +//# sourceMappingURL=Transformation.js.map \ No newline at end of file diff --git a/dist/src/Transformation.js.map b/dist/src/Transformation.js.map new file mode 100644 index 0000000..7577995 --- /dev/null +++ b/dist/src/Transformation.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Transformation.js","sourceRoot":"","sources":["../../src/Transformation.ts"],"names":[],"mappings":"AAiBA;;;GAGG;AACH,MAAM,OAAO,oBAAoB;IAG/B;;;;;;OAMG;IACH,YAAY,EAA2C;QACrD,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;IACf,CAAC;IAED;;;;;OAKG;IACH,SAAS,CAAC,KAAY,EAAE,MAAc;QACpC,OAAO,IAAI,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;IAChC,CAAC;CACF"} \ No newline at end of file diff --git a/dist/src/Universe.js b/dist/src/Universe.js new file mode 100644 index 0000000..f371dad --- /dev/null +++ b/dist/src/Universe.js @@ -0,0 +1,57 @@ +/** + * A Universe object that contains previous and current state of the universe, a simulation function, frame of reference transformations and other necessary data. + * @category Building blocks + */ +export class Universe { + /** + * Create a new Universe with the provided configuration. + * @param config configuration object. + */ + constructor(config) { + if (config.currState === undefined) + throw new Error('Missing Current State in Universe'); + if (config.simFunc === undefined) + throw new Error('Missing Simulation Function in Universe'); + this.label = config.label === undefined ? 'Universe' : config.label; + this.prevState + = config.prevState === undefined ? config.currState : config.prevState; + this.currState = config.currState; + this.color + = config.color === undefined ? 'rgba(255, 255, 255, 1)' : config.color; + this.simFunc = config.simFunc; + this.transformations + = config.transformations === undefined + ? [] + : Array.isArray(config.transformations) + ? config.transformations + : [config.transformations]; + } + /** + * Simulate a step in the Universe using the SimulateFunction and Transformations. + * @param deltaT time step to simulate. + * @hidden + */ + simulateStep(deltaT) { + let newState = this.simFunc.simulate(deltaT, this.currState, this.prevState); + this.prevState = this.currState; + this.transformations.forEach((t) => { + newState = t.transform(newState, deltaT); + }); + this.currState = newState; + } + /** + * Deep copy the current Universe. + * @returns a new Universe instance. + */ + clone() { + return new Universe({ + prevState: this.prevState.clone(), + currState: this.currState.clone(), + color: this.color, + label: this.label, + simFunc: this.simFunc, + transformations: this.transformations, + }); + } +} +//# sourceMappingURL=Universe.js.map \ No newline at end of file diff --git a/dist/src/Universe.js.map b/dist/src/Universe.js.map new file mode 100644 index 0000000..6801ab0 --- /dev/null +++ b/dist/src/Universe.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Universe.js","sourceRoot":"","sources":["../../src/Universe.ts"],"names":[],"mappings":"AAmCA;;;GAGG;AACH,MAAM,OAAO,QAAQ;IAoBnB;;;OAGG;IACH,YAAY,MAA+B;QACzC,IAAI,MAAM,CAAC,SAAS,KAAK,SAAS;YAAE,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;QACzF,IAAI,MAAM,CAAC,OAAO,KAAK,SAAS;YAAE,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;QAC7F,IAAI,CAAC,KAAK,GAAG,MAAM,CAAC,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;QACpE,IAAI,CAAC,SAAS;cACV,MAAM,CAAC,SAAS,KAAK,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC;QACzE,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;QAClC,IAAI,CAAC,KAAK;cACN,MAAM,CAAC,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,wBAAwB,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;QACzE,IAAI,CAAC,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC;QAC9B,IAAI,CAAC,eAAe;cAChB,MAAM,CAAC,eAAe,KAAK,SAAS;gBACpC,CAAC,CAAC,EAAE;gBACJ,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,eAAe,CAAC;oBACrC,CAAC,CAAC,MAAM,CAAC,eAAe;oBACxB,CAAC,CAAC,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;IACnC,CAAC;IAED;;;;OAIG;IACH,YAAY,CAAC,MAAc;QACzB,IAAI,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAClC,MAAM,EACN,IAAI,CAAC,SAAS,EACd,IAAI,CAAC,SAAS,CACf,CAAC;QACF,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,SAAS,CAAC;QAChC,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YACjC,QAAQ,GAAG,CAAC,CAAC,SAAS,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;QAC3C,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACH,KAAK;QACH,OAAO,IAAI,QAAQ,CAAC;YAClB,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE;YACjC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE;YACjC,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,eAAe,EAAE,IAAI,CAAC,eAAe;SACtC,CAAC,CAAC;IACL,CAAC;CACF"} \ No newline at end of file diff --git a/dist/src/Visualizer.js b/dist/src/Visualizer.js new file mode 100644 index 0000000..25785bb --- /dev/null +++ b/dist/src/Visualizer.js @@ -0,0 +1,2 @@ +export {}; +//# sourceMappingURL=Visualizer.js.map \ No newline at end of file diff --git a/dist/src/Visualizer.js.map b/dist/src/Visualizer.js.map new file mode 100644 index 0000000..71c014d --- /dev/null +++ b/dist/src/Visualizer.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Visualizer.js","sourceRoot":"","sources":["../../src/Visualizer.ts"],"names":[],"mappings":""} \ No newline at end of file diff --git a/dist/src/index.js b/dist/src/index.js new file mode 100644 index 0000000..1ed6e96 --- /dev/null +++ b/dist/src/index.js @@ -0,0 +1,18 @@ +/** + * NBody is a JS/TS library for simulating and visualizing n-body systems. It provides interfaces and implementations of forces, frame of reference transformations, simulation control, and multiple modes and paradigms of visualization. + * @author Yeluri Ketan + */ +import { CelestialBody, } from './CelestialBody'; +import { LambdaForce } from './Force'; +import { CentripetalForce, CombinedForce, Gravity, } from './library/Force'; +import { ExplicitEulerSim, RungeKutta4Sim, SemiImplicitEulerSim, VelocityVerletSim, } from './library/SimulateFunction'; +import { LambdaSim } from './SimulateFunction'; +import { Simulation } from './Simulation'; +import { State } from './State'; +import { BodyCenterTransformation, CoMTransformation, RotateTransformation, } from './library/Transformation'; +import { LambdaTransformation } from './Transformation'; +import { Universe } from './Universe'; +import { RealTimeVisualizer, RealTimeVisualizer3D, RecordingVisualizer, RecordingVisualizer3D, } from './library/Visualizer'; +import { Vector3 } from 'three'; +export { BodyCenterTransformation, CelestialBody, CentripetalForce, CombinedForce, CoMTransformation, ExplicitEulerSim, Gravity, LambdaForce, LambdaSim, LambdaTransformation, RealTimeVisualizer, RealTimeVisualizer3D, RecordingVisualizer, RecordingVisualizer3D, RotateTransformation, RungeKutta4Sim, SemiImplicitEulerSim, Simulation, State, Universe, Vector3, VelocityVerletSim, }; +//# sourceMappingURL=index.js.map \ No newline at end of file diff --git a/dist/src/index.js.map b/dist/src/index.js.map new file mode 100644 index 0000000..7e15b5c --- /dev/null +++ b/dist/src/index.js.map @@ -0,0 +1 @@ +{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EACL,aAAa,GACd,MAAM,iBAAiB,CAAC;AAEzB,OAAO,EAAE,WAAW,EAAc,MAAM,SAAS,CAAC;AAClD,OAAO,EACL,gBAAgB,EAAE,aAAa,EAAE,OAAO,GACzC,MAAM,iBAAiB,CAAC;AAEzB,OAAO,EACL,gBAAgB,EAAE,cAAc,EAChC,oBAAoB,EAAE,iBAAiB,GACxC,MAAM,4BAA4B,CAAC;AACpC,OAAO,EAAE,SAAS,EAAyB,MAAM,oBAAoB,CAAC;AAEtE,OAAO,EAAE,UAAU,EAAqC,MAAM,cAAc,CAAC;AAE7E,OAAO,EAAE,KAAK,EAAE,MAAM,SAAS,CAAC;AAEhC,OAAO,EACL,wBAAwB,EACxB,iBAAiB,EAAE,oBAAoB,GACxC,MAAM,0BAA0B,CAAC;AAClC,OAAO,EAAE,oBAAoB,EAAuB,MAAM,kBAAkB,CAAC;AAE7E,OAAO,EAAE,QAAQ,EAAuB,MAAM,YAAY,CAAC;AAE3D,OAAO,EACL,kBAAkB,EAAE,oBAAoB,EACxC,mBAAmB,EAAE,qBAAqB,GAC3C,MAAM,sBAAsB,CAAC;AAE9B,OAAO,EAAE,OAAO,EAAE,MAAM,OAAO,CAAC;AAEhC,OAAO,EACL,wBAAwB,EAAE,aAAa,EAAE,gBAAgB,EACzD,aAAa,EAAE,iBAAiB,EAAE,gBAAgB,EAAE,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,oBAAoB,EAAE,kBAAkB,EAC7H,oBAAoB,EACpB,mBAAmB,EACnB,qBAAqB,EACrB,oBAAoB,EAAE,cAAc,EACpC,oBAAoB,EAAE,UAAU,EAAE,KAAK,EAAE,QAAQ,EAAE,OAAO,EAAE,iBAAiB,GAC9E,CAAC"} \ No newline at end of file diff --git a/dist/src/library/Force.js b/dist/src/library/Force.js new file mode 100644 index 0000000..aa9d3ad --- /dev/null +++ b/dist/src/library/Force.js @@ -0,0 +1,105 @@ +import { Vector3 } from 'three'; +/** + * Represents a [Newtonian Gravitational force](https://en.wikipedia.org/wiki/Newton%27s_law_of_universal_gravitation) object. + * @category Forces + */ +export class Gravity { + /** + * Create a new Gravity with the provided gravitational constant. + * @param G gravitational constant. + */ + constructor(G = 6.674e-11) { + this.G = G; + } + /** + * Calculate and return the forces acting on the bodies. arr[i] represents the force acting on the ith body as a result of all other bodies. + * @param bodies celestial bodies + * @returns forces + */ + getForces(bodies) { + let n = bodies.length; + let ans = []; + for (let i = 0; i < n; i++) { + ans.push(new Vector3(0, 0, 0)); + } + for (let i = 0; i < n; i++) { + for (let j = i + 1; j < n; j++) { + let currForce = this.calcNewtonian(bodies[i], bodies[j]); + ans[i].add(currForce); + ans[j].sub(currForce); + } + } + return ans; + } + /** + * Calculate the gravitational force acting on body a due to body b. + * @param a body a. + * @param b body b. + * @returns gravitational force of a on b. + * @hidden + */ + calcNewtonian(a, b) { + let distSq = a.position.distanceToSquared(b.position); + let forceVal = (this.G * a.mass * b.mass) / distSq; + return b.position + .clone() + .sub(a.position) + .normalize() + .multiplyScalar(forceVal); + } +} +/** + * Represents a [Centripetal force](https://en.wikipedia.org/wiki/Centripetal_force) object. To be used to calculate the force required to keep the bodies in circular motion around a given center. + * @category Forces + */ +export class CentripetalForce { + /** + * Create a new CentripetalForce with the provided center of force. + * @param center center of force. + */ + constructor(center = new Vector3(0, 0, 0)) { + this.center = center; + } + /** + * Calculate the force required to keep the bodies in circular motion around the center. arr[i] represents the centripetal force required for the ith body. + * @param bodies celestial bodies. + * @returns forces. + */ + getForces(bodies) { + // TODO: mv^2/r sin or cos thetha? + return bodies.map((body) => { + const directionVector = this.center.clone() + .sub(body.position); + return directionVector.setLength((body.mass * body.velocity.lengthSq()) / directionVector.length()); + }); + } +} +/** + * Represents a combined force object. To be used to additively combine multiple forces acting on a system of bodies. + * @category Forces + */ +export class CombinedForce { + /** + * Create a new CombinedForce with the provided forces. + * @param forces array of forces. + */ + constructor(forces) { + this.forces = forces; + } + /** + * Get the combined forces acting on the bodies. arr[i] represents the combined force acting on the ith body as a result of all force systems. + * @param bodies celestial bodies. + * @returns element-wise combined forces. + */ + getForces(bodies) { + const forceVal = bodies.map(() => new Vector3(0, 0, 0)); + this.forces.forEach((force) => { + force.getForces(bodies) + .forEach((val, index) => { + forceVal[index].add(val); + }); + }); + return forceVal; + } +} +//# sourceMappingURL=Force.js.map \ No newline at end of file diff --git a/dist/src/library/Force.js.map b/dist/src/library/Force.js.map new file mode 100644 index 0000000..f7ad36c --- /dev/null +++ b/dist/src/library/Force.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Force.js","sourceRoot":"","sources":["../../../src/library/Force.ts"],"names":[],"mappings":"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"} \ No newline at end of file diff --git a/dist/src/library/SimulateFunction.js b/dist/src/library/SimulateFunction.js new file mode 100644 index 0000000..9b2dce7 --- /dev/null +++ b/dist/src/library/SimulateFunction.js @@ -0,0 +1,339 @@ +import { Vector3 } from 'three'; +import { State } from '../State'; +// export class VerletSim implements SimulateFunction { +// forceCalculator: Force; +// prevDeltaT: number | undefined = undefined; +// constructor(forceCalculator: Force) { +// this.forceCalculator = forceCalculator; +// } +// private fromOneState( +// currState: State, +// deltaT: number, +// forces: Vector3[] +// ): State { +// let updatedBodies: CelestialBody[] = currState.bodies.map((body, index) => { +// const currAccel = forces[index].divideScalar(body.mass); +// const newPos = currAccel +// .clone() +// .multiplyScalar(deltaT / 2) +// .add(body.velocity) +// .multiplyScalar(deltaT) +// .add(body.position); +// return body.clone( +// newPos, +// this.verletVel(body.position, newPos, deltaT), +// currAccel +// ); +// }); +// this.prevDeltaT = deltaT; +// return new State(updatedBodies); +// } +// simulate(deltaT: number, currState: State, prevState?: State): State { +// if (deltaT <= 0) { +// return currState.clone(); +// } +// let forces = this.forceCalculator.getForces(currState.bodies); +// if (forces.length !== currState.bodies.length) { +// console.error(`forces length !== number of bodies`); +// return currState.clone(); +// } +// if (prevState === undefined) { +// return this.fromOneState(currState, deltaT, forces); +// } +// let updatedBodies: CelestialBody[] = currState.bodies.map((body, index) => { +// const currAccel = forces[index].divideScalar(body.mass); +// const newPos = this.verletPos( +// prevState.bodies[index].position, +// body.position, +// currAccel, +// deltaT +// ); +// return body.clone( +// newPos, +// this.verletVel(body.position, newPos, deltaT), +// currAccel +// ); +// }); +// this.prevDeltaT = deltaT; +// return new State(updatedBodies); +// } +// verletPos( +// oldPos: Vector3, +// currPos: Vector3, +// currAccel: Vector3, +// deltaT: number +// ): Vector3 { +// const prevDT = this.prevDeltaT === undefined ? deltaT : this.prevDeltaT; +// return currAccel +// .clone() +// .multiplyScalar((prevDT + deltaT) / 2) +// .add(currPos.clone().sub(oldPos).divideScalar(prevDT)) +// .multiplyScalar(deltaT) +// .add(currPos); +// } +// verletVel(currPos: Vector3, newPos: Vector3, deltaT: number): Vector3 { +// return newPos.clone().sub(currPos).divideScalar(deltaT); +// } +// } +/** + * Represents a simulation function object that uses the [Velocity Verlet integration method](https://en.wikipedia.org/wiki/Verlet_integration#Velocity_Verlet) to simulate the motion of bodies. + * @category SimulateFunctions + */ +export class VelocityVerletSim { + /** + * Create a new VelocityVerletSim with the provided force calculator, which is invoked on every simulation step. + * @param forceCalculator force calculator. + */ + constructor(forceCalculator) { + this.forceCalculator = forceCalculator; + } + /** + * Simulate a step in the Universe by using the previous and/or current state and a time step, using the Velocity Verlet integration method. + * @param deltaT time step. + * @param currState current state. + * @returns new state after the simulation step. + */ + simulate(deltaT, currState) { + if (deltaT <= 0) { + return currState.clone(); + } + // x(t + dt) = x(t) + (v(t) * dt) + (0.5 * a(t) * dt * dt) + let updatedBodies = currState.bodies.map((v) => { + let updatedBody = v.clone(); + updatedBody.position = this.calcNewPos(updatedBody.position, updatedBody.velocity, updatedBody.acceleration, deltaT); + return updatedBody; + }); + // a(t + dt) = accel at x(t + dt) + let newForces = this.forceCalculator.getForces(updatedBodies); + return new State(updatedBodies.map((b, i) => { + let newAccel = newForces[i].divideScalar(b.mass); + // v(t + dt) = v(t) + 0.5 * (a(t) + a(t + dt)) * dt + b.velocity.add(b.acceleration.add(newAccel) + .multiplyScalar(deltaT / 2)); + b.acceleration = newAccel; + return b; + })); + } + /** + * Calculate the new position of a body in the following way: + * + * newPos = currPos + currVel * deltaT + 0.5 * currAccel * deltaT^2. + * @param currPos current position. + * @param currVel current velocity. + * @param currAccel current acceleration. + * @param deltaT time step. + * @returns new position. + * @hidden + */ + calcNewPos(currPos, currVel, currAccel, deltaT) { + return currPos + .clone() + .add(currVel.clone() + .multiplyScalar(deltaT)) + .add(currAccel.clone() + .multiplyScalar(deltaT * deltaT * 0.5)); + } +} +/** + * Represents a simulation function object that uses the [Euler integration method](https://en.wikipedia.org/wiki/Euler_method) to simulate motions of bodies. + * @category SimulateFunctions + */ +export class ExplicitEulerSim { + /** + * Create a new ExplicitEulerSim with the provided force calculator, which is invoked on every simulation step. + * @param force force calculator. + */ + constructor(force) { + this.force = force; + } + /** + * Simulate a step in the Universe by using the current state and a time step, using the Euler integration method. + * @param deltaT time step. + * @param currState current state. + * @returns new state after the simulation step. + */ + simulate(deltaT, currState) { + const updatedBodies = currState.bodies.map((b) => b.clone( + // x(t + dt) = x(t) + v(t) * dt + this.rateUpdate(b.position, b.velocity, deltaT), + // v(t + dt) = v(t) + a(t) * dt + this.rateUpdate(b.velocity, b.acceleration, deltaT))); + const updatedForces = this.force.getForces(updatedBodies); + updatedBodies.forEach((b, i) => { + // a(t + dt) = accel at x(t + dt) + b.acceleration = updatedForces[i].divideScalar(b.mass); + }); + return new State(updatedBodies); + } + /** + * Update a vector quantity using the rate of change and a time step. + * @param prev previous value. + * @param rate rate of change. + * @param deltaT time step. + * @returns updated value. + * @hidden + */ + rateUpdate(prev, rate, deltaT) { + return rate.clone() + .multiplyScalar(deltaT) + .add(prev); + } +} +/** + * Represents a simulation function object that uses the [Semi-Implicit Euler integration method](https://en.wikipedia.org/wiki/Semi-implicit_Euler_method) to simulate the motion of bodies. + * @category SimulateFunctions + */ +export class SemiImplicitEulerSim { + /** + * Create a new SemiImplicitEulerSim with the provided force calculator, which is invoked on every simulation step. + * @param force force calculator. + */ + constructor(force) { + this.force = force; + } + /** + * Simulate a step in the Universe by using the current state and a time step, using the Semi-Implicit Euler integration method. + * @param deltaT time step. + * @param currState current state. + * @returns new state after the simulation step. + */ + simulate(deltaT, currState) { + const updatedBodies = currState.bodies.map((b) => { + // v(t + dt) = v(t) + a(t) * dt + const updatedVel = this.rateUpdate(b.velocity, b.acceleration, deltaT); + return b.clone( + // x(t + dt) = x(t) + v(t + dt) * dt + this.rateUpdate(b.position, updatedVel, deltaT), updatedVel); + }); + const updatedForces = this.force.getForces(updatedBodies); + updatedBodies.forEach((b, i) => { + // a(t + dt) = accel at x(t + dt) + b.acceleration = updatedForces[i].divideScalar(b.mass); + }); + return new State(updatedBodies); + } + /** + * Update a vector quantity using the rate of change and a time step. + * @param prev previous value. + * @param rate rate of change. + * @param deltaT time step. + * @returns updated value. + * @hidden + */ + rateUpdate(prev, rate, deltaT) { + return rate.clone() + .multiplyScalar(deltaT) + .add(prev); + } +} +/** + * Represents a simulation function object that uses the [Runge-Kutta 4 integration method](https://en.wikipedia.org/wiki/Runge%E2%80%93Kutta_methods) to simulate the motion of bodies. + * @category SimulateFunctions + */ +export class RungeKutta4Sim { + /** + * Create a new RungeKutta4Sim with the provided weights for average force calculator, which is invoked on every simulation step. + * @param force force calculator. + * @param weights weights for weighted average. + */ + constructor(force, weights) { + this.force = force; + if (weights.length !== 4) { + throw new Error('Weights for RK4 must be of length 4'); + } + this.weights = weights; + } + /** + * Simulate a step in the Universe by using the current state and a time step, using the Runge-Kutta 4 integration method. + * @param deltaT time step. + * @param currState current state. + * @returns new state after the simulation step. + */ + simulate(deltaT, currState) { + let bodiesK = currState.bodies.map((v) => ({ + // k1v = a(x(t)) + kv: [v.acceleration.clone()], + // k1r = v(t) + kx: [v.velocity.clone()], + })); + // k2v = a(x(t) + k1x * dt / 2) + const k2v = this.getInterKV(currState.bodies, bodiesK, 0, deltaT / 2); + // k2x = v(t) + k1v * dt / 2 + const k2x = this.getInterKX(currState.bodies, bodiesK, 0, deltaT / 2); + bodiesK.forEach((v, i) => { + v.kv.push(k2v[i]); + v.kx.push(k2x[i]); + }); + // k3v = a(x(t) + k2x * dt / 2) + const k3v = this.getInterKV(currState.bodies, bodiesK, 1, deltaT / 2); + // k3x = v(t) + k2v * dt / 2 + const k3x = this.getInterKX(currState.bodies, bodiesK, 1, deltaT / 2); + bodiesK.forEach((v, i) => { + v.kv.push(k3v[i]); + v.kx.push(k3x[i]); + }); + // k3v = a(x(t) + k3x * dt) + const k4v = this.getInterKV(currState.bodies, bodiesK, 2, deltaT); + // k4x = v(t) + k3v * dt + const k4x = this.getInterKX(currState.bodies, bodiesK, 2, deltaT); + bodiesK.forEach((v, i) => { + v.kv.push(k4v[i]); + v.kx.push(k4x[i]); + }); + const updatedBodies = currState.bodies.map((b, i) => { + const weightedXRate = new Vector3(); + const weightedVRate = new Vector3(); + bodiesK[i].kx.forEach((v, j) => { + weightedXRate.add(v.multiplyScalar(this.weights[j])); + }); + bodiesK[i].kv.forEach((v, j) => { + weightedVRate.add(v.multiplyScalar(this.weights[j])); + }); + return b.clone(weightedXRate.multiplyScalar(deltaT / 6) + .add(b.position), weightedVRate.multiplyScalar(deltaT / 6) + .add(b.velocity)); + }); + const updatedForces = this.force.getForces(updatedBodies); + updatedBodies.forEach((v, i) => { + v.acceleration = updatedForces[i].divideScalar(v.mass); + }); + return new State(updatedBodies); + } + /** + * Get intermediate velocity rate values. + * @param bodies current bodies. + * @param bodiesK intermediate kinematic rate values for all bodies. + * @param krInd index of intermediate kinematic rate value to base the new intermediate value on. + * @param deltaT time step. + * @returns next intermediate velocity rate values for all bodies. + * @hidden + */ + getInterKV(bodies, bodiesK, krInd, deltaT) { + // newX = x(t) + k(kInd)x * delta + let newBodies = bodies.map((v, i) => { + let newBody = v.clone(); + newBody.position.add(bodiesK[i].kx[krInd].clone() + .multiplyScalar(deltaT)); + return newBody; + }); + // accel at newX + return this.force.getForces(newBodies) + .map((v, i) => v.divideScalar(bodies[i].mass)); + } + /** + * Get intermediate position rate values. + * @param bodies current bodies. + * @param bodiesK intermediate kinematic rate values for all bodies. + * @param kInd index of intermediate kinematic rate value to base the new intermediate value on. + * @param deltaT time step. + * @returns next intermediate position rate values for all bodies. + * @hidden + */ + getInterKX(bodies, bodiesK, kInd, deltaT) { + // v(t) + k(kInd)v * deltaT + return bodies.map((v, i) => v.velocity.clone() + .add(bodiesK[i].kv[kInd].clone() + .multiplyScalar(deltaT))); + } +} +//# sourceMappingURL=SimulateFunction.js.map \ No newline at end of file diff --git a/dist/src/library/SimulateFunction.js.map b/dist/src/library/SimulateFunction.js.map new file mode 100644 index 0000000..a491a01 --- /dev/null +++ b/dist/src/library/SimulateFunction.js.map @@ -0,0 +1 @@ +{"version":3,"file":"SimulateFunction.js","sourceRoot":"","sources":["../../../src/library/SimulateFunction.ts"],"names":[],"mappings":"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"} \ No newline at end of file diff --git a/dist/src/library/Transformation.js b/dist/src/library/Transformation.js new file mode 100644 index 0000000..ea1384a --- /dev/null +++ b/dist/src/library/Transformation.js @@ -0,0 +1,91 @@ +import { Vector3 } from 'three'; +/** + * Frame of reference transformation to the center of the first body in the system. + * @category Transformations + */ +export class BodyCenterTransformation { + /** + * Transform the frame of reference to the center of the first body in the system. + * @param state state to transform. + * @returns transformed state. + */ + transform(state) { + const transform = state.bodies[0].position.clone(); + state.bodies.forEach((b) => { + b.position.sub(transform); + }); + return state; + } +} +/** + * Frame of reference transformation to the center of mass of the system. + * @category Transformations + */ +export class CoMTransformation { + /** + * Transform the frame of reference to the center of mass of the system. + * @param state state to transform. + * @returns transformed state. + */ + transform(state) { + let totalMass = 0; + let com = new Vector3(); + state.bodies.forEach((b) => { + totalMass += b.mass; + com.add(b.position.clone() + .multiplyScalar(b.mass)); + }); + com.divideScalar(totalMass); + state.bodies.forEach((b) => { + b.position.sub(com); + }); + return state; + } +} +/** + * Frame of reference transformation around an axis by an angle. Makes sense to this transformation only during initialization of the universe and not at every time step. + * @category Transformations + */ +export class RotateTransformation { + /** + * Create a new RotateTransformer. + * @param axis axis to rotate around. + * @param angle angle to rotate by. + */ + constructor(axis, angle) { + this.axis = axis; + this.angle = angle; + } + /** + * Transform the frame of reference around an axis by an angle. + * @param state state to transform. + * @returns transformed state. + */ + transform(state) { + state.bodies.forEach((b) => { + b.position.applyAxisAngle(this.axis, this.angle); + b.velocity.applyAxisAngle(this.axis, this.angle); + b.acceleration.applyAxisAngle(this.axis, this.angle); + }); + return state; + } +} +// export class PinTransformer implements Transformer { +// readonly axis: Vector3; +// readonly index: number; +// constructor(axis: Vector3, index: number) { +// this.axis = axis; +// this.index = index +// } +// transform(state: State): State { +// const angle = state.bodies[this.index].position.clone().angleTo(this.axis); +// const pivot = state.bodies[this.index].position.clone().cross(this.axis.clone()).normalize(); +// state.bodies.forEach((b) => { +// b.position.applyAxisAngle(pivot.clone(), angle); +// b.velocity.applyAxisAngle(pivot.clone(), angle); +// b.acceleration.applyAxisAngle(pivot.clone(), angle); +// }); +// return state; +// } +// } +//# sourceMappingURL=Transformation.js.map \ No newline at end of file diff --git a/dist/src/library/Transformation.js.map b/dist/src/library/Transformation.js.map new file mode 100644 index 0000000..61b9676 --- /dev/null +++ b/dist/src/library/Transformation.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Transformation.js","sourceRoot":"","sources":["../../../src/library/Transformation.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,OAAO,CAAC;AAIhC;;;GAGG;AACH,MAAM,OAAO,wBAAwB;IACnC;;;;OAIG;IACH,SAAS,CAAC,KAAY;QACpB,MAAM,SAAS,GAAG,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;QACnD,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YACzB,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC5B,CAAC,CAAC,CAAC;QACH,OAAO,KAAK,CAAC;IACf,CAAC;CACF;AAED;;;GAGG;AACH,MAAM,OAAO,iBAAiB;IAC5B;;;;OAIG;IACH,SAAS,CAAC,KAAY;QACpB,IAAI,SAAS,GAAG,CAAC,CAAC;QAClB,IAAI,GAAG,GAAG,IAAI,OAAO,EAAE,CAAC;QACxB,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YACzB,SAAS,IAAI,CAAC,CAAC,IAAI,CAAC;YACpB,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,EAAE;iBACvB,cAAc,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;QAC5B,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YACzB,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACtB,CAAC,CAAC,CAAC;QACH,OAAO,KAAK,CAAC;IACf,CAAC;CACF;AAED;;;GAGG;AACH,MAAM,OAAO,oBAAoB;IAI/B;;;;OAIG;IACH,YAAY,IAAa,EAAE,KAAa;QACtC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,KAAK,GAAG,KAAK,CAAC;IACrB,CAAC;IAED;;;;OAIG;IACH,SAAS,CAAC,KAAY;QACpB,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE;YACzB,CAAC,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YACjD,CAAC,CAAC,QAAQ,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YACjD,CAAC,CAAC,YAAY,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QACvD,CAAC,CAAC,CAAC;QACH,OAAO,KAAK,CAAC;IACf,CAAC;CACF;AAED,uDAAuD;AACvD,4BAA4B;AAC5B,4BAA4B;AAE5B,gDAAgD;AAChD,wBAAwB;AACxB,yBAAyB;AACzB,MAAM;AAEN,qCAAqC;AACrC,kFAAkF;AAClF,oGAAoG;AACpG,oCAAoC;AACpC,yDAAyD;AACzD,yDAAyD;AACzD,6DAA6D;AAC7D,UAAU;AACV,oBAAoB;AACpB,MAAM;AACN,IAAI"} \ No newline at end of file diff --git a/dist/src/library/Visualizer.js b/dist/src/library/Visualizer.js new file mode 100644 index 0000000..01cfd2e --- /dev/null +++ b/dist/src/library/Visualizer.js @@ -0,0 +1,1035 @@ +import GUI from 'lil-gui'; +import Plotly from 'plotly.js-dist'; +import * as THREE from 'three'; +import { OrbitControls, ViewHelper } from 'three/examples/jsm/Addons'; +import Stats from 'three/examples/jsm/libs/stats.module'; +let animationId = null; +/** + * Clips a number to a minimum and maximum value. + * @param x number to clip. + * @param min minimum value. + * @param max maximum value. + * @returns clipped value. + */ +function clipMinMax(x, min, max) { + if (x < min) + return min; + if (x > max) + return max; + return x; +} +/** + * Container object for body trails in a 2D universe based in Plotly. + */ +class PlotlyUniverseTrail { + /** + * Constructor for PlotlyUniverseTrail + * @param maxTrailLength max number of trail points to keep. + * @param color color of the trail. + */ + constructor(maxTrailLength, color) { + this.data = { + x: [], + y: [], + mode: 'markers', + marker: { + size: 1, + color: 'white', + }, + }; + this.data.marker.color = color; + this.trailLength = 0; + this.maxTrailLength = maxTrailLength; + this.trailInd = 0; + } + /** + * Add a trail point to the trail data, or replace an existing trail point if the max trail length has been reached. + * @param x x position. + * @param y y position. + */ + addTrail(x, y) { + if (this.trailLength < this.maxTrailLength) { + this.data.x.push(x); + this.data.y.push(y); + this.trailLength++; + } + else { + this.data.x[this.trailInd] = x; + this.data.y[this.trailInd] = y; + this.trailInd = (this.trailInd + 1) % this.trailLength; + } + } + /** + * Pop all trail points from the trail data. + */ + popAllTrails() { + this.data.x = []; + this.data.y = []; + this.trailLength = 0; + this.trailInd = 0; + } +} +/** + * 2D real-time visualizer using Plotly. + * @category Visualizers + */ +export class RealTimeVisualizer { + /** + * Constructor for RealTimeVisualizer + * @param simulation simulation object + */ + constructor(simulation) { + this.divId = ''; + this.universeTrails = []; + this.simulation = simulation; + } + /** + * Adds default controls using lil-gui to the visualization. + * @param parentElement parent element to place the controller div in. + */ + addControls(parentElement) { + const gui = new GUI({ + container: parentElement, + }); + gui.domElement.style.position = 'absolute'; + gui.domElement.style.top = '0'; + gui.domElement.style.left = '0'; + gui.domElement.style.zIndex = '1000'; + const config = this.simulation.controls; + gui.add(config, 'speed'); + gui.add(config, 'showTrails') + .onChange((value) => { + if (value === false) { + this.universeTrails.forEach((ut) => ut.popAllTrails()); + } + config.showTrails = value; + }); + const showUniverseFolder = gui.addFolder('Show Universe'); + showUniverseFolder.open(false); + this.simulation.universes.forEach((u, i) => { + showUniverseFolder + .add(config.showUniverse, u.label) + .onChange((value) => { + if (value === false) { + this.universeTrails[i].popAllTrails(); + } + config.showUniverse[u.label] = value; + }); + }); + } + /** + * Simulate and play the visualization. + * @param divId div id to render the visualization in. + * @param width width of the visualization. + * @param height height of the visualization. + */ + start(divId, width, height) { + if (this.divId !== '') { + // throw new Error( + // 'Simulation already playing. Stop the current playtime before initiating a new one.', + // ); + console.error('Simulation already playing. Stop the current playtime before initiating a new one.'); + return; + } + this.divId = divId; + let element = document.getElementById(divId); + if (element === null) { + return; + } + // const width = element.clientWidth; + // const height = element.clientHeight; + let maxWidth = 0; + let maxHeight = 0; + this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => { + maxWidth = Math.max(maxWidth, Math.abs(b.position.x)); + maxHeight = Math.max(maxHeight, Math.abs(b.position.y)); + })); + const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth); + const layout = { + paper_bgcolor: '#000000', + plot_bgcolor: '#000000', + font: { + color: '#bfbfbf', + }, + xaxis: { + autorange: false, + range: [-(width / 2) / scale, width / 2 / scale], + }, + yaxis: { + autorange: false, + range: [-(height / 2) / scale, height / 2 / scale], + }, + // uirevision: 'true', + showlegend: false, + width, + height, + }; + if (this.simulation.controller === 'ui') { + this.addControls(element); + } + let stats; + if (this.simulation.showDebugInfo) { + stats = new Stats(); + stats.dom.style.position = 'absolute'; + stats.dom.style.bottom = '0px'; + stats.dom.style.removeProperty('top'); + element.appendChild(stats.dom); + } + const init_data = this.simulation.universes.flatMap((uni) => { + const currTrail = new PlotlyUniverseTrail(this.simulation.getMaxTrailLength(), typeof uni.color === 'string' ? uni.color : uni.color[0]); + this.universeTrails.push(currTrail); + const currData = { + x: uni.currState.bodies.map((body) => body.position.x), + y: uni.currState.bodies.map((body) => body.position.y), + type: 'scatter', + mode: 'markers', + marker: { + color: uni.color, + sizemin: 6, + size: uni.currState.bodies.map((body) => Math.min(10, body.mass)), + }, + }; + if (this.simulation.getShowTrails()) { + uni.currState.bodies.forEach((b) => { + currTrail.addTrail(b.position.x, b.position.y); + }); + return [currData, currTrail.data]; + } + return [ + currData, + { + x: [], + y: [], + }, + ]; + }); + Plotly.newPlot(divId, init_data, layout, { + scrollZoom: true, + modeBarButtonsToRemove: [ + 'lasso2d', + 'select2d', + 'toImage', + 'resetScale2d', + ], + }); + const timePerFrame = 1000 / this.simulation.maxFrameRate; + if (animationId !== null) + return; + let lastPaintTimestampMs = 0; + let lastSimTimestampMs = 0; + /** + * Simulate a step in the simulation + * @param timestampMs current timestamp in milliseconds, sourced from requestAnimationFrame + */ + const step = (timestampMs) => { + this.simulation.simulateStep((this.simulation.controls.speed + * Math.min(timestampMs - lastSimTimestampMs, 33.33)) + / 1000); + lastSimTimestampMs = timestampMs; + }; + /** + * Paint the visualization + * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame + */ + const paint = (timestampMs) => { + if (this.simulation.controls.speed === 0 + || this.simulation.controls.paused) { + animationId = requestAnimationFrame(paint); + return; + } + step(timestampMs); + if (timePerFrame > 0 + && timestampMs - lastPaintTimestampMs < timePerFrame) { + animationId = requestAnimationFrame(paint); + return; + } + lastPaintTimestampMs = timestampMs; + const new_data = this.simulation.universes.flatMap((uni, i) => { + if (!this.simulation.getShowUniverse(uni.label)) { + return [ + { + x: [], + y: [], + }, + {}, + ]; + } + const currData = { + x: uni.currState.bodies.map((body) => body.position.x), + y: uni.currState.bodies.map((body) => body.position.y), + hovertext: uni.currState.bodies.map((body) => body.label), + marker: { + size: uni.currState.bodies.map((body) => Math.min(10, body.mass)), + color: uni.color, + sizemin: 6, + }, + mode: 'markers', + }; + let trailData = {}; + if (this.simulation.getShowTrails()) { + const currTrail = this.universeTrails[i]; + uni.currState.bodies.forEach((b) => { + currTrail.addTrail(b.position.x, b.position.y); + }); + trailData = currTrail.data; + } + return [currData, trailData]; + }); + Plotly.react(divId, new_data, layout); + if (this.simulation.showDebugInfo && stats) { + stats.update(); + } + animationId = requestAnimationFrame(paint); + }; + animationId = requestAnimationFrame(paint); + } + /** + * Stop the simulation and visualization. + */ + stop() { + Plotly.purge(this.divId); + this.divId = ''; + this.universeTrails.forEach((ut) => { + ut.popAllTrails(); + }); + this.universeTrails = []; + } +} +/** + * Container object for body trails in a 3D universe based in Three.js. + */ +class ThreeUniverseTrail { + /** + * Constructor for ThreeUniverseTrail. + * @param maxTrailLength max number of trail points to keep. + * @param color color of the trace points. + * @param scene scene to add trail points object to. + * @param scale scale of the visualizationl, used to set the size of the trail point. + */ + constructor(maxTrailLength, color, scene, scale) { + const geometry = new THREE.BufferGeometry(); + geometry.setAttribute('position', new THREE.BufferAttribute(new Float32Array(0), 3)); + this.trails = new THREE.Points(geometry, new THREE.PointsMaterial({ + color, + size: 0.005 * scale, + })); + scene.add(this.trails); + this.trailInd = 0; + this.trailLength = 0; + this.maxTrailLength = maxTrailLength; + } + /** + * Add a trail point at the given position to the scene. Replace an existing trail point if the max trail length has been reached. + * @param pos position to add trace point at. + */ + addTrail(pos) { + if (this.trailLength < this.maxTrailLength) { + this.trails.visible = true; + this.trailLength++; + const posArray = new Float32Array(this.trailLength * 3); + posArray.set(this.trails.geometry.attributes.position.array); + posArray.set(pos.toArray(), this.trailLength * 3 - 3); + this.trails.geometry.setAttribute('position', new THREE.BufferAttribute(posArray, 3)); + this.trails.geometry.attributes.position.needsUpdate = true; + } + else { + this.trails.geometry.attributes.position.array.set(pos.toArray(), this.trailInd * 3); + this.trailInd = (this.trailInd + 1) % this.maxTrailLength; + this.trails.geometry.attributes.position.needsUpdate = true; + } + } + /** + * Pop all trail points. + */ + popAllTrails() { + this.trails.visible = false; + this.trails.geometry.setAttribute('position', new THREE.BufferAttribute(new Float32Array(0), 3)); + this.trailInd = 0; + this.trailLength = 0; + } +} +/** + * 3D real-time visualizer using Three.js. + * @category Visualizers + */ +export class RealTimeVisualizer3D { + /** + * Constructor for RealTimeVisualizer3D. + * @param simulation simulation object. + */ + constructor(simulation) { + this.universeTrails = []; + this.simulation = simulation; + } + /** + * Adds default controls to the visualization. + * @param parentElement parent element to place the controller div in. + */ + addControls(parentElement) { + const gui = new GUI({ + container: parentElement, + }); + gui.domElement.style.position = 'absolute'; + gui.domElement.style.top = '0'; + gui.domElement.style.left = '0'; + gui.domElement.style.zIndex = '1000'; + const config = this.simulation.controls; + gui.add(config, 'speed'); + gui.add(config, 'showTrails') + .onChange((value) => { + if (value === false) { + this.universeTrails.forEach((ut) => { + ut.popAllTrails(); + }); + } + config.showTrails = value; + }); + const showUniverseFolder = gui.addFolder('Show Universe'); + showUniverseFolder.open(false); + this.simulation.universes.forEach((u, i) => { + showUniverseFolder + .add(config.showUniverse, u.label) + .onChange((value) => { + if (value === false) { + this.universeTrails[i].popAllTrails(); + } + config.showUniverse[u.label] = value; + }); + }); + } + /** + * Simulate and play the visualization + * @param divId div id to render the visualization in + * @param width width of the visualization. + * @param height height of the visualization. + */ + start(divId, width, height) { + if (this.scene !== undefined) { + // throw new Error( + // 'Simulation already playing. Stop the current playtime before initiating a new one.', + // ); + console.error('Simulation already playing. Stop the current playtime before initiating a new one.'); + return; + } + let element = document.getElementById(divId); + if (element === null) { + return; + } + element.style.position = 'relative'; + // const width = element.clientWidth; + // const height = element.clientHeight; + let maxWidth = 0; + let maxHeight = 0; + this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => { + maxWidth = Math.max(maxWidth, Math.abs(b.position.x)); + maxHeight = Math.max(maxHeight, Math.abs(b.position.y)); + })); + const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth); + this.scene = new THREE.Scene(); + const camera = new THREE.OrthographicCamera(width / -2, width / 2, height / 2, height / -2, 0, 10000000000); + camera.position.set(0, 0, Math.max(width, height)); + const renderer = new THREE.WebGLRenderer(); + renderer.setSize(width, height); + renderer.autoClear = false; + element.appendChild(renderer.domElement); + let stats; + if (this.simulation.showDebugInfo) { + stats = new Stats(); + stats.dom.style.position = 'absolute'; + stats.dom.style.right = '0px'; + stats.dom.style.removeProperty('left'); + element.appendChild(stats.dom); + } + if (this.simulation.controller === 'ui') { + this.addControls(element); + } + // const earthDiv = document.createElement('div'); + // earthDiv.className = 'label'; + // earthDiv.textContent = 'Earthhgkjfdghkjfgh'; + // earthDiv.style.backgroundColor = 'transparent'; + // earthDiv.style.color = 'white'; + // earthDiv.style.fontFamily = 'sans-serif'; + // earthDiv.style.background = 'rgba(0, 0, 0, 0.6)'; + // const earthLabel = new CSS2DObject(earthDiv); + // earthLabel.position.set(0, 0, 0); + // earthLabel.center.set(0, 1); + // // arr[0].add(earthLabel); + // earthLabel.layers.set(0); + // const labelRenderer = new CSS2DRenderer(); + // labelRenderer.setSize(width, height); + // labelRenderer.domElement.style.position = 'absolute'; + // labelRenderer.domElement.style.top = '0px'; + // element.appendChild(labelRenderer.domElement); + const orbitControls = new OrbitControls(camera, renderer.domElement); + orbitControls.listenToKeyEvents(window); + orbitControls.update(); + const axesHelper = new THREE.AxesHelper(width); + this.scene.add(axesHelper); + const viewHelper = new ViewHelper(camera, renderer.domElement); + // var m: Map = new Map(); + let arr = []; + this.simulation.universes.forEach((u) => { + this.universeTrails.push(new ThreeUniverseTrail(this.simulation.maxTrailLength, typeof u.color === 'string' ? u.color : u.color[0], this.scene, scale)); + u.currState.bodies.forEach((b) => { + const sph = new THREE.SphereGeometry(clipMinMax(Math.log2(b.mass) - 70, 10, 40), 8, 8); + const curr = new THREE.WireframeGeometry(sph); + const line = new THREE.LineSegments(curr, new THREE.LineBasicMaterial({ + // @ts-ignore + color: new THREE.Color(u.color), + })); + this.scene.add(line); + line.position.copy(b.position.clone() + .multiplyScalar(scale)); + // m.set(u.label + " " + b.label, line); + arr.push(line); + }); + }); + // arr[0].add(earthLabel) + const timePerFrame = 1000 / this.simulation.maxFrameRate; + let lastSimTimestampMs = performance.now(); + let lastPaint = performance.now(); + /** + * Simulate a step in the simulation + * @param timestampMs current timestamp in milliseconds, sourced from requestAnimationFrame + */ + const step = (timestampMs) => { + this.simulation.simulateStep((this.simulation.controls.speed + * Math.min(timestampMs - lastSimTimestampMs, 16.67)) + / 1000); + lastSimTimestampMs = timestampMs; + }; + /** + * Paint the visualization + * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame + */ + const paint = (timestampMs) => { + if (this.simulation.controls.speed === 0 + || this.simulation.controls.paused) { + requestAnimationFrame(paint); + renderer.clear(); + renderer.render(this.scene, camera); + viewHelper.render(renderer); + // labelRenderer.render(scene, camera); + orbitControls.update(); + return; + } + step(timestampMs); + if (timePerFrame > 0 && timestampMs - lastPaint < timePerFrame) { + requestAnimationFrame(paint); + renderer.clear(); + renderer.render(this.scene, camera); + viewHelper.render(renderer); + // labelRenderer.render(scene, camera); + orbitControls.update(); + return; + } + lastPaint = timestampMs; + if (this.simulation.showDebugInfo && stats) { + stats.update(); + } + let ind = 0; + this.simulation.universes.forEach((u, i) => { + if (this.simulation.controls.showUniverse[u.label]) { + u.currState.bodies.forEach((b) => { + arr[ind].visible = true; + arr[ind].position.copy(b.position.clone() + .multiplyScalar(scale)); + if (this.simulation.controls.showTrails) { + this.universeTrails[i].addTrail(arr[ind].position); + } + ind++; + }); + } + else { + u.currState.bodies.forEach((b) => { + arr[ind].visible = false; + ind++; + }); + } + }); + requestAnimationFrame(paint); + renderer.clear(); + renderer.render(this.scene, camera); + viewHelper.render(renderer); + // labelRenderer.render(scene, camera); + orbitControls.update(); + }; + requestAnimationFrame(paint); + } + /** + * Stop the simulation and visualization. + */ + stop() { + var _a; + (_a = this.scene) === null || _a === void 0 ? void 0 : _a.clear(); + this.scene = undefined; + this.universeTrails.forEach((ut) => { + ut.popAllTrails(); + }); + this.universeTrails = []; + } +} +/** + * 2D recording visualizer using Plotly. + * @category Visualizers + */ +export class RecordingVisualizer { + /** + * Constructor for RealTimeVisualizer + * @param simulation simulation object + */ + constructor(simulation) { + this.divId = ''; + this.universeTrails = []; + this.simulation = simulation; + } + /** + * Adds default controls using lil-gui to the visualization. + * @param parentElement parent element to place the controller div in. + */ + addControls(parentElement) { + const gui = new GUI({ + container: parentElement, + }); + gui.domElement.style.position = 'absolute'; + gui.domElement.style.top = '0'; + gui.domElement.style.left = '0'; + gui.domElement.style.zIndex = '1000'; + const config = this.simulation.controls; + gui.add(config, 'speed'); + gui.add(config, 'showTrails') + .onChange((value) => { + if (value === false) { + this.universeTrails.forEach((ut) => ut.popAllTrails()); + } + config.showTrails = value; + }); + const showUniverseFolder = gui.addFolder('Show Universe'); + showUniverseFolder.open(false); + this.simulation.universes.forEach((u, i) => { + showUniverseFolder + .add(config.showUniverse, u.label) + .onChange((value) => { + if (value === false) { + this.universeTrails[i].popAllTrails(); + } + config.showUniverse[u.label] = value; + }); + }); + } + /** + * Simulate and play the visualization. + * @param divId div id to render the visualization in. + * @param width width of the visualization. + * @param height height of the visualization. + * @param recordFor number of seconds to record for.. + */ + start(divId, width, height, recordFor) { + if (this.divId !== '') { + // throw new Error( + // 'Simulation already playing. Stop the current playtime before initiating a new one.', + // ); + console.error('Simulation already playing. Stop the current playtime before initiating a new one.'); + return; + } + this.divId = divId; + let element = document.getElementById(divId); + if (element === null) { + return; + } + // const width = element.clientWidth; + // const height = element.clientHeight; + let maxWidth = 0; + let maxHeight = 0; + this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => { + maxWidth = Math.max(maxWidth, Math.abs(b.position.x)); + maxHeight = Math.max(maxHeight, Math.abs(b.position.y)); + })); + const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth); + const recordedFrames = []; + const totalFrames = this.simulation.maxFrameRate * recordFor; + let playInd = 1; + this.simulation.universes.forEach((u) => { + recordedFrames.push([u.currState.clone()]); + }); + for (let i = 0; i < totalFrames; i++) { + this.simulation.simulateStep(1 / this.simulation.maxFrameRate); + this.simulation.universes.forEach((u, j) => { + recordedFrames[j].push(u.currState.clone()); + }); + } + const layout = { + paper_bgcolor: '#000000', + plot_bgcolor: '#000000', + font: { + color: '#bfbfbf', + }, + xaxis: { + autorange: false, + range: [-(width / 2) / scale, width / 2 / scale], + }, + yaxis: { + autorange: false, + range: [-(height / 2) / scale, height / 2 / scale], + }, + // uirevision: 'true', + showlegend: false, + width, + height, + }; + if (this.simulation.controller === 'ui') { + this.addControls(element); + } + let stats; + if (this.simulation.showDebugInfo) { + stats = new Stats(); + stats.dom.style.position = 'absolute'; + stats.dom.style.bottom = '0px'; + stats.dom.style.removeProperty('top'); + element.appendChild(stats.dom); + } + const init_data = this.simulation.universes.flatMap((uni) => { + const currTrail = new PlotlyUniverseTrail(this.simulation.getMaxTrailLength(), typeof uni.color === 'string' ? uni.color : uni.color[0]); + this.universeTrails.push(currTrail); + const currData = { + x: uni.currState.bodies.map((body) => body.position.x), + y: uni.currState.bodies.map((body) => body.position.y), + type: 'scatter', + mode: 'markers', + marker: { + color: uni.color, + sizemin: 6, + size: uni.currState.bodies.map((body) => Math.min(10, body.mass)), + }, + }; + if (this.simulation.getShowTrails()) { + uni.currState.bodies.forEach((b) => { + currTrail.addTrail(b.position.x, b.position.y); + }); + return [currData, currTrail.data]; + } + return [ + currData, + { + x: [], + y: [], + }, + ]; + }); + Plotly.newPlot(divId, init_data, layout, { + scrollZoom: true, + modeBarButtonsToRemove: [ + 'zoom2d', + 'lasso2d', + 'select2d', + 'toImage', + 'resetScale2d', + ], + }); + if (animationId !== null) + return; + /** + * Paint the visualization + * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame + */ + const paint = (timestampMs) => { + if (this.simulation.controls.speed === 0 + || this.simulation.controls.paused) { + animationId = requestAnimationFrame(paint); + return; + } + const currPlayInd = Math.round(playInd); + const new_data = this.simulation.universes.flatMap((uni, i) => { + if (!this.simulation.getShowUniverse(uni.label)) { + return [ + { + x: [], + y: [], + }, + {}, + ]; + } + const currState = recordedFrames[i][currPlayInd]; + const currData = { + x: currState.bodies.map((body) => body.position.x), + y: currState.bodies.map((body) => body.position.y), + hovertext: currState.bodies.map((body) => body.label), + marker: { + size: currState.bodies.map((body) => Math.min(10, body.mass)), + color: uni.color, + sizemin: 6, + }, + mode: 'markers', + }; + let trailData = {}; + if (this.simulation.getShowTrails()) { + const currTrail = this.universeTrails[i]; + currState.bodies.forEach((b) => { + currTrail.addTrail(b.position.x, b.position.y); + }); + trailData = currTrail.data; + } + return [currData, trailData]; + }); + Plotly.react(divId, new_data, layout); + if (this.simulation.showDebugInfo && stats) { + stats.update(); + } + playInd = Math.round(playInd + this.simulation.controls.speed); + if (playInd < 0) { + if (this.simulation.looped) { + playInd = ((playInd % totalFrames) + totalFrames) % totalFrames; + } + else { + playInd = 0; + } + } + else if (playInd >= totalFrames) { + if (this.simulation.looped) { + playInd %= totalFrames; + } + else { + playInd = totalFrames - 1; + } + } + animationId = requestAnimationFrame(paint); + }; + animationId = requestAnimationFrame(paint); + } + /** + * Stop the simulation and visualization. + */ + stop() { + Plotly.purge(this.divId); + this.divId = ''; + this.universeTrails = []; + } +} +/** + * 3D recording visualizer using Three.js. + * @category Visualizers + */ +export class RecordingVisualizer3D { + /** + * Constructor for RealTimeVisualizer3D. + * @param simulation simulation object. + */ + constructor(simulation) { + this.universeTrails = []; + this.simulation = simulation; + } + /** + * Adds default controls to the visualization. + * @param parentElement parent element to place the controller div in. + */ + addControls(parentElement) { + const gui = new GUI({ + container: parentElement, + }); + gui.domElement.style.position = 'absolute'; + gui.domElement.style.top = '0'; + gui.domElement.style.left = '0'; + gui.domElement.style.zIndex = '1000'; + const config = this.simulation.controls; + gui.add(config, 'speed'); + gui.add(config, 'showTrails') + .onChange((value) => { + if (value === false) { + this.universeTrails.forEach((ut) => { + ut.popAllTrails(); + }); + } + config.showTrails = value; + }); + const showUniverseFolder = gui.addFolder('Show Universe'); + showUniverseFolder.open(false); + this.simulation.universes.forEach((u, i) => { + showUniverseFolder + .add(config.showUniverse, u.label) + .onChange((value) => { + if (value === false) { + this.universeTrails[i].popAllTrails(); + } + config.showUniverse[u.label] = value; + }); + }); + } + /** + * Simulate and play the visualization + * @param divId div id to render the visualization in. + * @param width width of the visualization. + * @param height height of the visualization. + * @param recordFor number of seconds to record for. + */ + start(divId, width, height, recordFor) { + if (this.scene !== undefined) { + // throw new Error( + // 'Simulation already playing. Stop the current playtime before initiating a new one.', + // ); + console.error('Simulation already playing. Stop the current playtime before initiating a new one.'); + return; + } + let element = document.getElementById(divId); + if (element === null) { + return; + } + // const width = element.clientWidth; + // const height = element.clientHeight; + let maxWidth = 0; + let maxHeight = 0; + this.simulation.universes.forEach((u) => u.currState.bodies.forEach((b) => { + maxWidth = Math.max(maxWidth, Math.abs(b.position.x)); + maxHeight = Math.max(maxHeight, Math.abs(b.position.y)); + })); + const scale = 0.5 * Math.min(height / maxHeight, width / maxWidth); + this.scene = new THREE.Scene(); + const camera = new THREE.OrthographicCamera(width / -2, width / 2, height / 2, height / -2, 0, 10000000000); + camera.position.set(0, 0, Math.max(width, height)); + const renderer = new THREE.WebGLRenderer(); + renderer.setSize(width, height); + renderer.autoClear = false; + element.appendChild(renderer.domElement); + let stats; + if (this.simulation.showDebugInfo) { + stats = new Stats(); + stats.dom.style.position = 'absolute'; + stats.dom.style.right = '0px'; + stats.dom.style.removeProperty('left'); + element.appendChild(stats.dom); + } + if (this.simulation.controller === 'ui') { + this.addControls(element); + } + // const earthDiv = document.createElement('div'); + // earthDiv.className = 'label'; + // earthDiv.textContent = 'Earthhgkjfdghkjfgh'; + // earthDiv.style.backgroundColor = 'transparent'; + // earthDiv.style.color = 'white'; + // earthDiv.style.fontFamily = 'sans-serif'; + // earthDiv.style.background = 'rgba(0, 0, 0, 0.6)'; + // const earthLabel = new CSS2DObject(earthDiv); + // earthLabel.position.set(0, 0, 0); + // earthLabel.center.set(0, 1); + // // arr[0].add(earthLabel); + // earthLabel.layers.set(0); + // const labelRenderer = new CSS2DRenderer(); + // labelRenderer.setSize(width, height); + // labelRenderer.domElement.style.position = 'absolute'; + // labelRenderer.domElement.style.top = '0px'; + // element.appendChild(labelRenderer.domElement); + const orbitControls = new OrbitControls(camera, renderer.domElement); + orbitControls.listenToKeyEvents(window); + orbitControls.update(); + const axesHelper = new THREE.AxesHelper(width); + this.scene.add(axesHelper); + const viewHelper = new ViewHelper(camera, renderer.domElement); + // var m: Map = new Map(); + let arr = []; + this.simulation.universes.forEach((u) => { + this.universeTrails.push(new ThreeUniverseTrail(this.simulation.maxTrailLength, typeof u.color === 'string' ? u.color : u.color[0], this.scene, scale)); + u.currState.bodies.forEach((b) => { + const sph = new THREE.SphereGeometry(clipMinMax(Math.log2(b.mass) - 70, 10, 40), 8, 8); + const curr = new THREE.WireframeGeometry(sph); + const line = new THREE.LineSegments(curr, new THREE.LineBasicMaterial({ + // @ts-ignore + color: new THREE.Color(u.color), + })); + this.scene.add(line); + line.position.copy(b.position.clone() + .multiplyScalar(scale)); + // m.set(u.label + " " + b.label, line); + arr.push(line); + }); + }); + // arr[0].add(earthLabel) + const recordedFrames = []; + const totalFrames = this.simulation.maxFrameRate * recordFor; + let playInd = 1; + this.simulation.universes.forEach((u) => { + recordedFrames.push([u.currState.clone()]); + }); + for (let i = 0; i < totalFrames; i++) { + this.simulation.simulateStep(1 / this.simulation.maxFrameRate); + this.simulation.universes.forEach((u, j) => { + recordedFrames[j].push(u.currState.clone()); + }); + } + /** + * Paint the visualization + * @param timestampMs current timestamp in milliseconds, provided by requestAnimationFrame + */ + const paint = (timestampMs) => { + if (this.simulation.controls.speed === 0 + || this.simulation.controls.paused) { + requestAnimationFrame(paint); + renderer.clear(); + renderer.render(this.scene, camera); + viewHelper.render(renderer); + // labelRenderer.render(scene, camera); + orbitControls.update(); + return; + } + let ind = 0; + this.simulation.universes.forEach((u, i) => { + if (this.simulation.controls.showUniverse[u.label]) { + const currState = recordedFrames[i][playInd]; + currState.bodies.forEach((b) => { + arr[ind].visible = true; + arr[ind].position.copy(b.position.clone() + .multiplyScalar(scale)); + if (this.simulation.controls.showTrails) { + this.universeTrails[i].addTrail(arr[ind].position); + } + ind++; + }); + } + else { + u.currState.bodies.forEach(() => { + arr[ind].visible = false; + ind++; + }); + } + }); + if (this.simulation.showDebugInfo && stats) { + stats.update(); + } + playInd = Math.round(playInd + this.simulation.controls.speed); + if (playInd < 0) { + if (this.simulation.looped) { + playInd = ((playInd % totalFrames) + totalFrames) % totalFrames; + } + else { + playInd = 0; + } + } + else if (playInd >= totalFrames) { + if (this.simulation.looped) { + playInd %= totalFrames; + } + else { + playInd = totalFrames - 1; + } + } + requestAnimationFrame(paint); + renderer.clear(); + renderer.render(this.scene, camera); + viewHelper.render(renderer); + // labelRenderer.render(scene, camera); + orbitControls.update(); + }; + requestAnimationFrame(paint); + } + /** + * Stop the simulation and visualization. + */ + stop() { + var _a; + (_a = this.scene) === null || _a === void 0 ? void 0 : _a.clear(); + this.scene = undefined; + this.universeTrails.forEach((ut) => { + ut.popAllTrails(); + }); + this.universeTrails = []; + } +} +//# sourceMappingURL=Visualizer.js.map \ No newline at end of file diff --git a/dist/src/library/Visualizer.js.map b/dist/src/library/Visualizer.js.map new file mode 100644 index 0000000..4d59043 --- /dev/null +++ b/dist/src/library/Visualizer.js.map @@ -0,0 +1 @@ +{"version":3,"file":"Visualizer.js","sourceRoot":"","sources":["../../../src/library/Visualizer.ts"],"names":[],"mappings":"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"} \ No newline at end of file diff --git a/dist/src/plotly.js b/dist/src/plotly.js new file mode 100644 index 0000000..49893ec --- /dev/null +++ b/dist/src/plotly.js @@ -0,0 +1,31 @@ +/** + * + */ +export class SimulationPlot { + /** + * Create a new SimulationPlot with the provided draw function, data, simulation, and layout. + * @param plotlyDrawFn draw function. + * @param data data to draw. + * @param sim simulation. + * @param layout layout of the plot. + */ + constructor(plotlyDrawFn, data, sim, layout) { + /** + * Returns a string representation of the object. + * @returns string representation of the object. + */ + this.toReplString = () => ''; + /** + * Draw the plot in the provided div. + * @param divId div id. + */ + this.draw = (divId) => { + this.plotlyDrawFn(divId, this.data, this.sim, this.layout); + }; + this.plotlyDrawFn = plotlyDrawFn; + this.data = data; + this.sim = sim; + this.layout = layout; + } +} +//# sourceMappingURL=plotly.js.map \ No newline at end of file diff --git a/dist/src/plotly.js.map b/dist/src/plotly.js.map new file mode 100644 index 0000000..72e1dcc --- /dev/null +++ b/dist/src/plotly.js.map @@ -0,0 +1 @@ +{"version":3,"file":"plotly.js","sourceRoot":"","sources":["../../src/plotly.ts"],"names":[],"mappings":"AAOA;;GAEG;AACH,MAAM,OAAO,cAAc;IAKzB;;;;;;OAMG;IACH,YACE,YAAiB,EACjB,IAAU,EACV,GAAe,EACf,MAAuB;QAOzB;;;WAGG;QACI,iBAAY,GAAG,GAAG,EAAE,CAAC,kBAAkB,CAAC;QAE/C;;;WAGG;QACI,SAAI,GAAG,CAAC,KAAa,EAAE,EAAE;YAC9B,IAAI,CAAC,YAAY,CACf,KAAK,EACL,IAAI,CAAC,IAAI,EACT,IAAI,CAAC,GAAG,EACR,IAAI,CAAC,MAAM,CACZ,CAAC;QACJ,CAAC,CAAC;QAtBA,IAAI,CAAC,YAAY,GAAG,YAAY,CAAC;QACjC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,GAAG,GAAG,GAAG,CAAC;QACf,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;CAmBF"} \ No newline at end of file diff --git a/dist/src/CelestialBody.d.ts b/dist/types/src/CelestialBody.d.ts similarity index 100% rename from dist/src/CelestialBody.d.ts rename to dist/types/src/CelestialBody.d.ts diff --git a/dist/src/Force.d.ts b/dist/types/src/Force.d.ts similarity index 100% rename from dist/src/Force.d.ts rename to dist/types/src/Force.d.ts diff --git a/dist/src/SimulateFunction.d.ts b/dist/types/src/SimulateFunction.d.ts similarity index 100% rename from dist/src/SimulateFunction.d.ts rename to dist/types/src/SimulateFunction.d.ts diff --git a/dist/src/Simulation.d.ts b/dist/types/src/Simulation.d.ts similarity index 100% rename from dist/src/Simulation.d.ts rename to dist/types/src/Simulation.d.ts diff --git a/dist/src/State.d.ts b/dist/types/src/State.d.ts similarity index 100% rename from dist/src/State.d.ts rename to dist/types/src/State.d.ts diff --git a/dist/src/Transformation.d.ts b/dist/types/src/Transformation.d.ts similarity index 100% rename from dist/src/Transformation.d.ts rename to dist/types/src/Transformation.d.ts diff --git a/dist/src/Universe.d.ts b/dist/types/src/Universe.d.ts similarity index 100% rename from dist/src/Universe.d.ts rename to dist/types/src/Universe.d.ts diff --git a/dist/src/Visualizer.d.ts b/dist/types/src/Visualizer.d.ts similarity index 100% rename from dist/src/Visualizer.d.ts rename to dist/types/src/Visualizer.d.ts diff --git a/dist/src/index.d.ts b/dist/types/src/index.d.ts similarity index 74% rename from dist/src/index.d.ts rename to dist/types/src/index.d.ts index 2c66886..4e74496 100644 --- a/dist/src/index.d.ts +++ b/dist/types/src/index.d.ts @@ -3,7 +3,7 @@ * @author Yeluri Ketan */ import { CelestialBody } from './CelestialBody'; -import { type Force } from './Force'; +import { LambdaForce, type Force } from './Force'; import { CentripetalForce, CombinedForce, Gravity } from './library/Force'; import { ExplicitEulerSim, RungeKutta4Sim, SemiImplicitEulerSim, VelocityVerletSim } from './library/SimulateFunction'; import { LambdaSim, type SimulateFunction } from './SimulateFunction'; @@ -14,4 +14,4 @@ import { LambdaTransformation, type Transformation } from './Transformation'; import { Universe, type UniverseConfig } from './Universe'; import { RealTimeVisualizer, RealTimeVisualizer3D, RecordingVisualizer, RecordingVisualizer3D } from './library/Visualizer'; import { Vector3 } from 'three'; -export { BodyCenterTransformation, CelestialBody, CentripetalForce, CombinedForce, CoMTransformation, ExplicitEulerSim, Gravity, LambdaSim, LambdaTransformation, RealTimeVisualizer, RealTimeVisualizer3D, RecordingVisualizer, RecordingVisualizer3D, RotateTransformation, RungeKutta4Sim, SemiImplicitEulerSim, Simulation, State, Universe, Vector3, VelocityVerletSim, type ControllerType, type Force, type SimulateFunction, type Transformation, type UniverseConfig, type VisType, }; +export { BodyCenterTransformation, CelestialBody, CentripetalForce, CombinedForce, CoMTransformation, ExplicitEulerSim, Gravity, LambdaForce, LambdaSim, LambdaTransformation, RealTimeVisualizer, RealTimeVisualizer3D, RecordingVisualizer, RecordingVisualizer3D, RotateTransformation, RungeKutta4Sim, SemiImplicitEulerSim, Simulation, State, Universe, Vector3, VelocityVerletSim, type ControllerType, type Force, type SimulateFunction, type Transformation, type UniverseConfig, type VisType, }; diff --git a/dist/src/library/Force.d.ts b/dist/types/src/library/Force.d.ts similarity index 100% rename from dist/src/library/Force.d.ts rename to dist/types/src/library/Force.d.ts diff --git a/dist/src/library/SimulateFunction.d.ts b/dist/types/src/library/SimulateFunction.d.ts similarity index 100% rename from dist/src/library/SimulateFunction.d.ts rename to dist/types/src/library/SimulateFunction.d.ts diff --git a/dist/src/library/Transformation.d.ts b/dist/types/src/library/Transformation.d.ts similarity index 100% rename from dist/src/library/Transformation.d.ts rename to dist/types/src/library/Transformation.d.ts diff --git a/dist/src/library/Visualizer.d.ts b/dist/types/src/library/Visualizer.d.ts similarity index 100% rename from dist/src/library/Visualizer.d.ts rename to dist/types/src/library/Visualizer.d.ts diff --git a/dist/src/plotly.d.ts b/dist/types/src/plotly.d.ts similarity index 100% rename from dist/src/plotly.d.ts rename to dist/types/src/plotly.d.ts diff --git a/docs/assets/navigation.js b/docs/assets/navigation.js index d3eac64..033ebc0 100644 --- a/docs/assets/navigation.js +++ b/docs/assets/navigation.js @@ -1 +1 @@ -window.navigationData = "data:application/octet-stream;base64,H4sIAAAAAAAACq2WS1PbMBSF/0pHa0NKAm3xkvAYhnYDaTaMF4p8k9xBj4x0zSTt5L93TENiWw/DwNbnnE9Xjyv58S8jWBPL2UWFskS9+DKTRjw5ljGxRFla0Cx/3LvGIMERcnlhyg3L2IrTkuVMSO4cuEFLPl6SkixjT6hLlp8Mf2yzPecBVSU5odE+5KAlCcQJAuH6cyr3W+MzWBeIviqp9BQEGTvywzvBy2b/HSxn5MojdEewJrCaS7YtGtxbTWDnXEBs4a+NFY2ace8fvCjtcYdn3/y1hutKi/aKNyhdUwo4sVy7ubGKR3Ftiw9rzv1lAvEDp8niCojLzhIczlzbkdq+sVEz1FDGUE05xbmx/Bkp0AA7IZV9904Wib2MLdvVeiVRIF1VEuwDKr/SriNV8k+uZiUPYvZSKn9f6QXcVUT8NAhp68muB4W3qm9uIVe6q6URSJspWAkURHqWN9xuH+24ItpzsW2vr926H8DGevR1PjFnunt+9WE9S/+x6kOGXMnDZuonoI8acqWon3znTTar+JVnNFkjJdjadRiK6sygrXZG+Xr+/eRsGHjtxkbPcdFltdU+1hRdqKDd53C6aOcrLvEP2NjM74HLCSo4OAMb53mSh8Fzjy7fwhxdpqnC2PpnKV2oZ3onM1xqwOZzi23xD10a3TfbCQAA" \ No newline at end of file +window.navigationData = "data:application/octet-stream;base64,H4sIAAAAAAAACq2WTU/bQBCG/0q1Z0NKAm3xkfAhRHuBNBfkw2Y9SUbsR7Q7Rkmr/PfKNCS298MguPp95/HM7szYj38ZwZpYzi4qlCXqxZeZNOLJsYyJJcrSgmb54941BgmOkMsLU25YxlaclixnQnLnwA1a8vGSlGQZe0Jdsvxk+GOb7TkPqCrJCY32IQctSSBOEAiuH6fifmt8BusCoa9KKnoKgowd+cE7wYvN/jtYzsiVR+iOYE1gNZdsWzS4t5rAzrmA2MFfGysaOePeP3hR2u8dnn3zzxquKy3aJ96gdE0p4MRy7ebGKh7FtS0+rFn7SwHxhtNkcQXEZecIDj3XdqSub2zUDDWUMVRTTnFuLH9GCgzATkjF/uRqVvJIBg0xxXh3NxSJfogd/dV6JVEgXVUS7AMqP9uuo7/sIGYvpeLvK72Au4qInwYhbT25OUDhreqrLeRKbwZpBNJmClYCBZGe5Q0b8qNTW0TnNnbt9equZwpsbM5f64k50xP4qw/rWfrbqg8ZciWbzdSfkT5qyJWifvLenGxW8bVpNFkjJdjadXgV1TGDttp5y9fz7ydnw8AXc2z0HBddVlvtY03RhRLaPQ5HF+34ikv8AzZW+T1wOUEFB2fg4jxPshk89+jyLczRZZoqjK1/uNKJeqZ3MsOpBmw+t9gW/wBFz1T7HwoAAA==" \ No newline at end of file diff --git a/docs/assets/search.js b/docs/assets/search.js index b3369ca..6aaa028 100644 --- a/docs/assets/search.js +++ b/docs/assets/search.js @@ -1 +1 @@ -window.searchData = "data:application/octet-stream;base64,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"; \ No newline at end of file +window.searchData = "data:application/octet-stream;base64,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"; \ No newline at end of file diff --git a/docs/classes/BodyCenterTransformation.html b/docs/classes/BodyCenterTransformation.html index b688e25..25ba4ce 100644 --- a/docs/classes/BodyCenterTransformation.html +++ b/docs/classes/BodyCenterTransformation.html @@ -1,7 +1,7 @@ BodyCenterTransformation | nbody

Class BodyCenterTransformation

Frame of reference transformation to the center of the first body in the system.

-

Implements

Constructors

Implements

Constructors

Methods

Constructors

Methods

Generated using TypeDoc

\ No newline at end of file +

Generated using TypeDoc

\ No newline at end of file diff --git a/docs/classes/CelestialBody.html b/docs/classes/CelestialBody.html index e86e342..f9178a5 100644 --- a/docs/classes/CelestialBody.html +++ b/docs/classes/CelestialBody.html @@ -1,5 +1,5 @@ CelestialBody | nbody

Class CelestialBody

Represents a celestial body with all of its kinematic properties.

-

Constructors

Constructors

Properties

acceleration label mass @@ -12,14 +12,14 @@
  • position: Vector3

    position of the body.

  • velocity: Vector3

    velocity of the body.

  • acceleration: Vector3

    acceleration of the body.

    -
  • Returns CelestialBody

    Properties

    acceleration: Vector3

    Acceleration vector of the body.

    -
    label: string

    Label of the body.

    -
    mass: number

    Mean mass of the body.

    -
    position: Vector3

    Position vector of the body.

    -
    velocity: Vector3

    Velocity vector of the body.

    -

    Methods

    • Deep copy the current CelestialBody with the updated kinematic properties.

      +

    Returns CelestialBody

    Properties

    acceleration: Vector3

    Acceleration vector of the body.

    +
    label: string

    Label of the body.

    +
    mass: number

    Mean mass of the body.

    +
    position: Vector3

    Position vector of the body.

    +
    velocity: Vector3

    Velocity vector of the body.

    +

    Methods

    • Deep copy the current CelestialBody with the updated kinematic properties.

      Parameters

      • Optional position: Vector3

        new position.

      • Optional velocity: Vector3

        new velocity.

      • Optional acceleration: Vector3

        new acceleration.

      Returns CelestialBody

      a new CelestialBody instance with the updated properties.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/CentripetalForce.html b/docs/classes/CentripetalForce.html index 01c5551..9b348a4 100644 --- a/docs/classes/CentripetalForce.html +++ b/docs/classes/CentripetalForce.html @@ -1,11 +1,11 @@ CentripetalForce | nbody

    Class CentripetalForce

    Represents a Centripetal force object. To be used to calculate the force required to keep the bodies in circular motion around a given center.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    Properties

    center: Vector3

    Center of force.

    -

    Methods

    • Calculate the force required to keep the bodies in circular motion around the center. arr[i] represents the centripetal force required for the ith body.

      +

    Returns CentripetalForce

    Properties

    center: Vector3

    Center of force.

    +

    Methods

    • Calculate the force required to keep the bodies in circular motion around the center. arr[i] represents the centripetal force required for the ith body.

      Parameters

      Returns Vector3[]

      forces.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/CoMTransformation.html b/docs/classes/CoMTransformation.html index a24514e..ecdeec4 100644 --- a/docs/classes/CoMTransformation.html +++ b/docs/classes/CoMTransformation.html @@ -1,7 +1,7 @@ CoMTransformation | nbody

    Class CoMTransformation

    Frame of reference transformation to the center of mass of the system.

    -

    Implements

    Constructors

    Implements

    Constructors

    Methods

    Constructors

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/CombinedForce.html b/docs/classes/CombinedForce.html index 0829071..19f1167 100644 --- a/docs/classes/CombinedForce.html +++ b/docs/classes/CombinedForce.html @@ -1,10 +1,10 @@ CombinedForce | nbody

    Class CombinedForce

    Represents a combined force object. To be used to additively combine multiple forces acting on a system of bodies.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    Properties

    forces: Force[]

    Methods

    • Get the combined forces acting on the bodies. arr[i] represents the combined force acting on the ith body as a result of all force systems.

      +

    Returns CombinedForce

    Properties

    forces: Force[]

    Methods

    • Get the combined forces acting on the bodies. arr[i] represents the combined force acting on the ith body as a result of all force systems.

      Parameters

      Returns Vector3[]

      element-wise combined forces.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/ExplicitEulerSim.html b/docs/classes/ExplicitEulerSim.html index 8d19f6a..5dd09fb 100644 --- a/docs/classes/ExplicitEulerSim.html +++ b/docs/classes/ExplicitEulerSim.html @@ -1,12 +1,12 @@ ExplicitEulerSim | nbody

    Class ExplicitEulerSim

    Represents a simulation function object that uses the Euler integration method to simulate motions of bodies.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    Properties

    force: Force

    Force object to calculate forces on bodies in the Universe.

    -

    Methods

    • Simulate a step in the Universe by using the current state and a time step, using the Euler integration method.

      +

    Returns ExplicitEulerSim

    Properties

    force: Force

    Force object to calculate forces on bodies in the Universe.

    +

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/Gravity.html b/docs/classes/Gravity.html index 58b90e3..928c57e 100644 --- a/docs/classes/Gravity.html +++ b/docs/classes/Gravity.html @@ -1,13 +1,13 @@ Gravity | nbody

    Class Gravity

    Represents a Newtonian Gravitational force object.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    G

    Methods

    Constructors

    • Create a new Gravity with the provided gravitational constant.

      Parameters

      • G: number = 6.674e-11

        gravitational constant.

        -

      Returns Gravity

    Properties

    G: number

    Gravitational constant.

    +

    Returns Gravity

    Properties

    G: number

    Gravitational constant.

    Default Value

    6.674e-11
     
    -

    Methods

    • Calculate and return the forces acting on the bodies. arr[i] represents the force acting on the ith body as a result of all other bodies.

      +

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/LambdaForce.html b/docs/classes/LambdaForce.html new file mode 100644 index 0000000..a039183 --- /dev/null +++ b/docs/classes/LambdaForce.html @@ -0,0 +1,16 @@ +LambdaForce | nbody

    Class LambdaForce

    Function object that uses the user-defined lambda function to calculate the forces acting on the bodies.

    +

    Implements

    Constructors

    Properties

    fn +

    Methods

    Constructors

    • Create a new LambdaForce with the provided lambda function.

      +
        +
      • Lambda function should take in an array of CelestialBodies and return an array of forces acting on the bodies.
      • +
      • arr[i] should represent the force acting on the ith body.
      • +
      • Length of the returned array should be equal to the length of the input array of CelestialBodies.
      • +
      +

      Parameters

      Returns LambdaForce

    Properties

    fn: ((bodies) => Vector3[])

    Lambda function to calculate forces, provided by the user.

    +

    Type declaration

    Methods

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/LambdaSim.html b/docs/classes/LambdaSim.html index 619babc..98ff811 100644 --- a/docs/classes/LambdaSim.html +++ b/docs/classes/LambdaSim.html @@ -1,5 +1,5 @@ LambdaSim | nbody

    Class LambdaSim

    Function object that uses the user-defined lambda function to simulate the Universe.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    • Create a new LambdaSim with the provided lambda function.

      @@ -8,9 +8,9 @@
    • The lambda function should call or calculate the forces action on the bodies by itself.

    Parameters

    • fn: ((deltaT, currState, prevState) => State)

      lambda function.

      -
        • (deltaT, currState, prevState): State
        • Parameters

          Returns State

    Returns LambdaSim

    Properties

    fn: ((deltaT, currState, prevState) => State)

    Type declaration

      • (deltaT, currState, prevState): State
      • Parameters

        Returns State

    Methods

    • Simulate the Universe using the lambda function.

      +
        • (deltaT, currState, prevState): State
        • Parameters

          Returns State

    Returns LambdaSim

    Properties

    fn: ((deltaT, currState, prevState) => State)

    Type declaration

      • (deltaT, currState, prevState): State
      • Parameters

        Returns State

    Methods

    • Simulate the Universe using the lambda function.

      Parameters

      • deltaT: number

        time step.

      • currState: State

        current state of the Universe.

      • prevState: State

        previous state of the Universe.

      Returns State

      the next state of the Universe.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/LambdaTransformation.html b/docs/classes/LambdaTransformation.html index 9b16155..cbbfac5 100644 --- a/docs/classes/LambdaTransformation.html +++ b/docs/classes/LambdaTransformation.html @@ -1,5 +1,5 @@ LambdaTransformation | nbody

    Class LambdaTransformation

    A Frame of Reference transformation that uses the user-defined lambda function.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    • Create a new LambdaTransformer with the provided lambda function.

      @@ -9,8 +9,8 @@
    • Transformed state should contain the same number of bodies as the input state, and the order should be preserved.

    Parameters

    • fn: ((state, deltaT) => State)

      lambda function.

      -

    Returns LambdaTransformation

    Properties

    fn: ((state, deltaT) => State)

    Type declaration

    Methods

    • Transform the state's frame of reference using the lambda function.

      +

    Returns LambdaTransformation

    Properties

    fn: ((state, deltaT) => State)

    Type declaration

    Methods

    • Transform the state's frame of reference using the lambda function.

      Parameters

      • state: State

        state to transform.

      • deltaT: number

        time step taken to get to this state. Only applicable for time-dependent transformations.

      Returns State

      transformed state.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/RealTimeVisualizer.html b/docs/classes/RealTimeVisualizer.html index 3bbbfee..f2d749c 100644 --- a/docs/classes/RealTimeVisualizer.html +++ b/docs/classes/RealTimeVisualizer.html @@ -1,5 +1,5 @@ RealTimeVisualizer | nbody

    Class RealTimeVisualizer

    2D real-time visualizer using Plotly.

    -

    Implements

    • Visualizer

    Constructors

    Implements

    • Visualizer

    Constructors

    Properties

    divId simulation universeTrails @@ -8,11 +8,11 @@ stop

    Constructors

    Properties

    divId: string = ''
    simulation: Simulation
    universeTrails: PlotlyUniverseTrail[] = []

    Methods

    • Adds default controls using lil-gui to the visualization.

      +

    Returns RealTimeVisualizer

    Properties

    divId: string = ''
    simulation: Simulation
    universeTrails: PlotlyUniverseTrail[] = []

    Methods

    • Adds default controls using lil-gui to the visualization.

      Parameters

      • parentElement: HTMLElement

        parent element to place the controller div in.

        -

      Returns void

    • Simulate and play the visualization.

      +

    Returns void

    • Simulate and play the visualization.

      Parameters

      • divId: string

        div id to render the visualization in.

      • width: number

        width of the visualization.

      • height: number

        height of the visualization.

        -

      Returns void

    Generated using TypeDoc

    \ No newline at end of file +

    Returns void

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/RealTimeVisualizer3D.html b/docs/classes/RealTimeVisualizer3D.html index 75c0a42..ad82f40 100644 --- a/docs/classes/RealTimeVisualizer3D.html +++ b/docs/classes/RealTimeVisualizer3D.html @@ -1,5 +1,5 @@ RealTimeVisualizer3D | nbody

    Class RealTimeVisualizer3D

    3D real-time visualizer using Three.js.

    -

    Implements

    • Visualizer

    Constructors

    Implements

    • Visualizer

    Constructors

    Properties

    Constructors

    Properties

    scene?: Scene
    simulation: Simulation
    universeTrails: ThreeUniverseTrail[] = []

    Methods

    • Adds default controls to the visualization.

      +

    Returns RealTimeVisualizer3D

    Properties

    scene?: Scene
    simulation: Simulation
    universeTrails: ThreeUniverseTrail[] = []

    Methods

    • Adds default controls to the visualization.

      Parameters

      • parentElement: HTMLElement

        parent element to place the controller div in.

        -

      Returns void

    • Simulate and play the visualization

      +

    Returns void

    • Simulate and play the visualization

      Parameters

      • divId: string

        div id to render the visualization in

      • width: number

        width of the visualization.

      • height: number

        height of the visualization.

        -

      Returns void

    Generated using TypeDoc

    \ No newline at end of file +

    Returns void

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/RecordingVisualizer.html b/docs/classes/RecordingVisualizer.html index 5b2c89f..3f2cef6 100644 --- a/docs/classes/RecordingVisualizer.html +++ b/docs/classes/RecordingVisualizer.html @@ -1,5 +1,5 @@ RecordingVisualizer | nbody

    Class RecordingVisualizer

    2D recording visualizer using Plotly.

    -

    Implements

    • Visualizer

    Constructors

    Implements

    • Visualizer

    Constructors

    Properties

    divId simulation universeTrails @@ -8,12 +8,12 @@ stop

    Constructors

    Properties

    divId: string = ''
    simulation: Simulation
    universeTrails: PlotlyUniverseTrail[] = []

    Methods

    • Adds default controls using lil-gui to the visualization.

      +

    Returns RecordingVisualizer

    Properties

    divId: string = ''
    simulation: Simulation
    universeTrails: PlotlyUniverseTrail[] = []

    Methods

    • Adds default controls using lil-gui to the visualization.

      Parameters

      • parentElement: HTMLElement

        parent element to place the controller div in.

        -

      Returns void

    • Simulate and play the visualization.

      +

    Returns void

    • Simulate and play the visualization.

      Parameters

      • divId: string

        div id to render the visualization in.

      • width: number

        width of the visualization.

      • height: number

        height of the visualization.

      • recordFor: number

        number of seconds to record for..

        -

      Returns void

    Generated using TypeDoc

    \ No newline at end of file +

    Returns void

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/RecordingVisualizer3D.html b/docs/classes/RecordingVisualizer3D.html index f1ee977..7e456ad 100644 --- a/docs/classes/RecordingVisualizer3D.html +++ b/docs/classes/RecordingVisualizer3D.html @@ -1,5 +1,5 @@ RecordingVisualizer3D | nbody

    Class RecordingVisualizer3D

    3D recording visualizer using Three.js.

    -

    Implements

    • Visualizer

    Constructors

    Implements

    • Visualizer

    Constructors

    Properties

    Constructors

    Properties

    scene?: Scene
    simulation: Simulation
    universeTrails: ThreeUniverseTrail[] = []

    Methods

    • Adds default controls to the visualization.

      +

    Returns RecordingVisualizer3D

    Properties

    scene?: Scene
    simulation: Simulation
    universeTrails: ThreeUniverseTrail[] = []

    Methods

    • Adds default controls to the visualization.

      Parameters

      • parentElement: HTMLElement

        parent element to place the controller div in.

        -

      Returns void

    • Simulate and play the visualization

      +

    Returns void

    • Simulate and play the visualization

      Parameters

      • divId: string

        div id to render the visualization in.

      • width: number

        width of the visualization.

      • height: number

        height of the visualization.

      • recordFor: number

        number of seconds to record for.

        -

      Returns void

    Generated using TypeDoc

    \ No newline at end of file +

    Returns void

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/RotateTransformation.html b/docs/classes/RotateTransformation.html index d4f098f..44a2d1b 100644 --- a/docs/classes/RotateTransformation.html +++ b/docs/classes/RotateTransformation.html @@ -1,12 +1,12 @@ RotateTransformation | nbody

    Class RotateTransformation

    Frame of reference transformation around an axis by an angle. Makes sense to this transformation only during initialization of the universe and not at every time step.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    Properties

    angle: number
    axis: Vector3

    Methods

    • Transform the frame of reference around an axis by an angle.

      +

    Returns RotateTransformation

    Properties

    angle: number
    axis: Vector3

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/RungeKutta4Sim.html b/docs/classes/RungeKutta4Sim.html index 3b0efd3..baff168 100644 --- a/docs/classes/RungeKutta4Sim.html +++ b/docs/classes/RungeKutta4Sim.html @@ -1,15 +1,15 @@ RungeKutta4Sim | nbody

    Class RungeKutta4Sim

    Represents a simulation function object that uses the Runge-Kutta 4 integration method to simulate the motion of bodies.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    Properties

    force: Force

    Force object to calculate forces on bodies in the Universe.

    -
    weights: number[]

    Weights for weighted average.

    -

    Methods

    • Simulate a step in the Universe by using the current state and a time step, using the Runge-Kutta 4 integration method.

      +

    Returns RungeKutta4Sim

    Properties

    force: Force

    Force object to calculate forces on bodies in the Universe.

    +
    weights: number[]

    Weights for weighted average.

    +

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/SemiImplicitEulerSim.html b/docs/classes/SemiImplicitEulerSim.html index e023c00..98284a4 100644 --- a/docs/classes/SemiImplicitEulerSim.html +++ b/docs/classes/SemiImplicitEulerSim.html @@ -1,12 +1,12 @@ SemiImplicitEulerSim | nbody

    Class SemiImplicitEulerSim

    Represents a simulation function object that uses the Semi-Implicit Euler integration method to simulate the motion of bodies.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    Properties

    force: Force

    Force object to calculate forces on bodies in the Universe.

    -

    Methods

    • Simulate a step in the Universe by using the current state and a time step, using the Semi-Implicit Euler integration method.

      +

    Returns SemiImplicitEulerSim

    Properties

    force: Force

    Force object to calculate forces on bodies in the Universe.

    +

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/Simulation.html b/docs/classes/Simulation.html index 8d57734..b817a07 100644 --- a/docs/classes/Simulation.html +++ b/docs/classes/Simulation.html @@ -1,5 +1,5 @@ Simulation | nbody

    Class Simulation

    A Simulation object that contains Universes and a Visualizer.

    -

    Constructors

    Constructors

    Methods

    Constructors

    • Create a new Simulation object with the provided Universes and visualization config.

      Parameters

      • universes: Universe | Universe[]

        array of Universes.

        -
      • __namedParameters: {
            controller?: ControllerType;
            looped?: boolean;
            maxFrameRate?: number;
            maxTrailLength?: number;
            record?: boolean;
            showDebugInfo?: boolean;
            showTrails?: boolean;
            visType?: VisType;
        }
        • Optional controller?: ControllerType
        • Optional looped?: boolean
        • Optional maxFrameRate?: number
        • Optional maxTrailLength?: number
        • Optional record?: boolean
        • Optional showDebugInfo?: boolean
        • Optional showTrails?: boolean
        • Optional visType?: VisType

      Returns Simulation

    Methods

    • Get the maximum trail length used in the visualization.

      +
    • __namedParameters: {
          controller?: ControllerType;
          looped?: boolean;
          maxFrameRate?: number;
          maxTrailLength?: number;
          record?: boolean;
          showDebugInfo?: boolean;
          showTrails?: boolean;
          visType?: VisType;
      }
      • Optional controller?: ControllerType
      • Optional looped?: boolean
      • Optional maxFrameRate?: number
      • Optional maxTrailLength?: number
      • Optional record?: boolean
      • Optional showDebugInfo?: boolean
      • Optional showTrails?: boolean
      • Optional visType?: VisType

    Returns Simulation

    Methods

    • Get the maximum trail length used in the visualization.

      Returns number

      maximum trail length.

      -
    • Get whether trails are shown in the visualization.

      +
    • Get whether trails are shown in the visualization.

      Returns boolean

      true if trails are shown.

      -
    • True if the universe with the given label is shown.

      +
    • True if the universe with the given label is shown.

      Parameters

      • label: string

        universe label.

      Returns boolean

      whether the universe is shown.

      -
    • Get the speed of the simulation.

      Returns number

      speed of the simulation as a scale of normal time.

      -
    • Get whether the simulation is playing.

      Returns boolean

      true if the simulation is playing.

      -
    • Pause the simulation. Only works if the controller is 'code'.

      -

      Returns void

    • Resume the simulation. Only works if the controller is 'code'.

      -

      Returns void

    • Set the maximum trail length used in the visualization. Changes only apply on the next Simulation.play() call.

      +
    • Pause the simulation. Only works if the controller is 'code'.

      +

      Returns void

    • Resume the simulation. Only works if the controller is 'code'.

      +

      Returns void

    • Set the maximum trail length used in the visualization. Changes only apply on the next Simulation.play() call.

      Parameters

      • maxTrailLength: number

        maximum trail length.

        -

      Returns void

    • Set whether to show trails in the visualization. Only works if the controller is 'code'.

      +

    Returns void

    • Set whether to show trails in the visualization. Only works if the controller is 'code'.

      Parameters

      • showTrails: boolean

        true to show trails.

        -

      Returns void

    • Set whether to show the universe with the given label. Only works if the controller is 'code'.

      +

    Returns void

    • Set whether to show the universe with the given label. Only works if the controller is 'code'.

      Parameters

      • label: string

        universe label.

      • show: boolean

        true to show the universe.

        -

      Returns void

    • Set the speed of the simulation. Only works if the controller is 'code'.

      +

    Returns void

    • Set the speed of the simulation. Only works if the controller is 'code'.

      Parameters

      • speed: number

        speed of the simulation as a scale of normal time.

        -

      Returns void

    • Insert the simulation visualization in the div with the given id.

      +

    Returns void

    • Insert the simulation visualization in the div with the given id.

      Parameters

      • divId: string

        div id.

      • width: number
      • height: number
      • speed: number = 1

        initial time scale.

      • paused: boolean = false

        whether to start the simulation paused.

      • recordFor: number = 0

        number of seconds to record for, only used if in record mode.

        -

      Returns void

    Generated using TypeDoc

    \ No newline at end of file +

    Returns void

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/State.html b/docs/classes/State.html index 020b955..9d082a7 100644 --- a/docs/classes/State.html +++ b/docs/classes/State.html @@ -1,10 +1,10 @@ State | nbody

    Class State

    Represents a Universe's state snapshot.

    -

    Constructors

    Constructors

    Properties

    Methods

    Constructors

    Properties

    bodies: CelestialBody[]

    Array of celestial bodies that make up this state of the Universe.

    -

    Methods

    • Deep copy this state

      +

    Returns State

    Properties

    bodies: CelestialBody[]

    Array of celestial bodies that make up this state of the Universe.

    +

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/Universe.html b/docs/classes/Universe.html index 5fa2c80..2a15817 100644 --- a/docs/classes/Universe.html +++ b/docs/classes/Universe.html @@ -1,5 +1,5 @@ Universe | nbody

    Class Universe

    A Universe object that contains previous and current state of the universe, a simulation function, frame of reference transformations and other necessary data.

    -

    Constructors

    Constructors

    Properties

    color currState label @@ -9,10 +9,10 @@

    Methods

    Constructors

    Properties

    color: string | string[]

    Color of the bodies in the Universe. A single color applied to all bodies or an array of colors applied to each body respectively. Incase of array, length should match the number of bodies in the state.

    -
    currState: State
    label: string

    Label of the Universe.

    -
    prevState: State

    Simulation function used to simulate the Universe.

    -
    transformations: Transformation[]

    Array of transformations to be applied to the Universe's state after simulation and before visualization.

    -

    Methods

    • Deep copy the current Universe.

      +

    Returns Universe

    Properties

    color: string | string[]

    Color of the bodies in the Universe. A single color applied to all bodies or an array of colors applied to each body respectively. Incase of array, length should match the number of bodies in the state.

    +
    currState: State
    label: string

    Label of the Universe.

    +
    prevState: State

    Simulation function used to simulate the Universe.

    +
    transformations: Transformation[]

    Array of transformations to be applied to the Universe's state after simulation and before visualization.

    +

    Methods

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/Vector3.html b/docs/classes/Vector3.html index 0bb5a71..18b0aed 100644 --- a/docs/classes/Vector3.html +++ b/docs/classes/Vector3.html @@ -1,8 +1,8 @@ -Vector3 | nbody

    Class Vector3

    3D vector. ( class Vector3 implements Vector )

    +Vector3 | nbody

    Class Vector3

    Example

    const a = new THREE.Vector3( 1, 0, 0 );
    const b = new THREE.Vector3( 0, 1, 0 );
    const c = new THREE.Vector3();
    c.crossVectors( a, b );
    -

    Implements

    • Vector

    Constructors

    Constructors

    • Parameters

      • Optional x: number
      • Optional y: number
      • Optional z: number

      Returns Vector3

    Properties

    isVector3: true
    x: number

    Default

    0
    +

    Constructors

    • Parameters

      • Optional x: number
      • Optional y: number
      • Optional z: number

      Returns Vector3

    Properties

    isVector3: true
    x: number

    Default

    0
     
    -
    y: number

    Default

    0
    +
    y: number

    Default

    0
     
    -
    z: number

    Default

    0
    +
    z: number

    Default

    0
     
    -

    Methods

    • Iterating through a Vector3 instance will yield its components (x, y, z) in the corresponding order.

      -

      Returns Iterator<number, any, undefined>

    • Adds v to this vector.

      -

      Parameters

      Returns this

    • Parameters

      • s: number

      Returns this

    • Parameters

      Returns this

    • Sets this vector to a + b.

      -

      Parameters

      Returns this

    • Parameters

      Returns number

    • Parameters

      Returns this

    • Parameters

      • euler: Euler

      Returns this

    • Parameters

      • m: Matrix3

      Returns this

    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • m: Matrix3

      Returns this

    • Parameters

      • q: Quaternion

      Returns this

    • Returns this

    • Parameters

      Returns this

    • Parameters

      • min: number
      • max: number

      Returns this

    • Parameters

      • min: number
      • max: number

      Returns this

    • Clones this vector.

      -

      Returns this

    • Copies value of v to this vector.

      -

      Parameters

      Returns this

    • Sets this vector to cross product of itself and v.

      -

      Parameters

      Returns this

    • Sets this vector to cross product of a and b.

      -

      Parameters

      Returns this

    • Computes distance of this vector to v.

      -

      Parameters

      Returns number

    • Parameters

      Returns number

      Deprecated

      Use .manhattanDistanceTo() instead.

      -
    • Computes squared distance of this vector to v.

      -

      Parameters

      Returns number

    • Parameters

      Returns this

    • Divides this vector by scalar s. +

    Methods

    • Iterating through a Vector3 instance will yield its components (x, y, z) in the corresponding order.

      +

      Returns Iterator<number, any, undefined>

    • Adds v to this vector.

      +

      Parameters

      • v: Vector3Like

      Returns this

    • Parameters

      • s: number

      Returns this

    • Parameters

      Returns this

    • Sets this vector to a + b.

      +

      Parameters

      • a: Vector3Like
      • b: Vector3Like

      Returns this

    • Parameters

      Returns number

    • Parameters

      Returns this

    • Parameters

      • euler: Euler

      Returns this

    • Parameters

      • m: Matrix3

      Returns this

    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • m: Matrix3

      Returns this

    • Parameters

      • q: QuaternionLike

      Returns this

    • Returns this

    • Parameters

      • min: Vector3Like
      • max: Vector3Like

      Returns this

    • Parameters

      • min: number
      • max: number

      Returns this

    • Parameters

      • min: number
      • max: number

      Returns this

    • Clones this vector.

      +

      Returns this

    • Copies value of v to this vector.

      +

      Parameters

      • v: Vector3Like

      Returns this

    • Sets this vector to cross product of itself and v.

      +

      Parameters

      • a: Vector3Like

      Returns this

    • Sets this vector to cross product of a and b.

      +

      Parameters

      • a: Vector3Like
      • b: Vector3Like

      Returns this

    • Computes distance of this vector to v.

      +

      Parameters

      • v: Vector3Like

      Returns number

    • Computes squared distance of this vector to v.

      +

      Parameters

      • v: Vector3Like

      Returns number

    • Parameters

      • v: Vector3Like

      Returns this

    • Divides this vector by scalar s. Set vector to ( 0, 0, 0 ) if s == 0.

      -

      Parameters

      • s: number

      Returns this

    • Computes dot product of this vector and v.

      -

      Parameters

      Returns number

    • Checks for strict equality of this vector and v.

      -

      Parameters

      Returns boolean

    • Returns this

    • Sets this vector's x, y and z value from the provided array or array-like.

      +

      Parameters

      • s: number

      Returns this

    • Computes dot product of this vector and v.

      +

      Parameters

      • v: Vector3Like

      Returns number

    • Checks for strict equality of this vector and v.

      +

      Parameters

      • v: Vector3Like

      Returns boolean

    • Returns this

    • Sets this vector's x, y and z value from the provided array or array-like.

      Parameters

      • array: number[] | ArrayLike<number>

        the source array or array-like.

      • Optional offset: number

        (optional) offset into the array. Default is 0.

        -

      Returns this

    • Parameters

      • attribute: BufferAttribute | InterleavedBufferAttribute
      • index: number

      Returns this

    • Parameters

      • index: number

      Returns number

    • Computes length of this vector.

      -

      Returns number

    • Computes squared length of this vector.

      -

      Returns number

    • Parameters

      Returns this

    • Parameters

      Returns this

    • Computes the Manhattan length (distance) from this vector to the given vector v

      +

    Returns this

    • Parameters

      • attribute: BufferAttribute | InterleavedBufferAttribute
      • index: number

      Returns this

    • Parameters

      • index: number

      Returns number

    • Computes length of this vector.

      +

      Returns number

    • Computes squared length of this vector.

      +

      Returns number

    • Parameters

      • v: Vector3Like
      • alpha: number

      Returns this

    • Parameters

      • v1: Vector3Like
      • v2: Vector3Like
      • alpha: number

      Returns this

    • Computes the Manhattan length (distance) from this vector to the given vector v

      see Taxicab Geometry

      -

      Parameters

      Returns number

    • Computes the Manhattan length of this vector.

      +

      Parameters

      • v: Vector3Like

      Returns number

    • Computes the Manhattan length of this vector.

      see Taxicab Geometry

      -

      Returns number

    • Parameters

      Returns this

    • Parameters

      Returns this

    • Parameters

      Returns this

    • Multiplies this vector by scalar s.

      -

      Parameters

      • s: number

      Returns this

    • Parameters

      Returns this

    • Inverts this vector.

      -

      Returns this

    • Normalizes this vector.

      -

      Returns this

    • Parameters

      • camera: Camera

      Returns this

    • Parameters

      Returns this

    • Parameters

      Returns this

    • Sets this vector's x, y and z from Math.random

      -

      Returns this

    • Returns this

    • Parameters

      Returns this

    • Returns this

    • Returns this

    • Sets value of this vector.

      -

      Parameters

      • x: number
      • y: number
      • z: number

      Returns this

    • Parameters

      • index: number
      • value: number

      Returns this

    • Sets this vector's x, y and z components from the r, g, and b components of the specified +

      Returns number

    • Parameters

      • v: Vector3Like

      Returns this

    • Parameters

      • v: Vector3Like

      Returns this

    • Parameters

      • v: Vector3Like

      Returns this

    • Multiplies this vector by scalar s.

      +

      Parameters

      • s: number

      Returns this

    • Parameters

      • a: Vector3Like
      • b: Vector3Like

      Returns this

    • Inverts this vector.

      +

      Returns this

    • Normalizes this vector.

      +

      Returns this

    • Parameters

      • camera: Camera

      Returns this

    • Parameters

      Returns this

    • Parameters

      Returns this

    • Sets this vector's x, y and z from Math.random

      +

      Returns this

    • Returns this

    • Parameters

      • vector: Vector3Like

      Returns this

    • Returns this

    • Returns this

    • Sets value of this vector.

      +

      Parameters

      • x: number
      • y: number
      • z: number

      Returns this

    • Parameters

      • index: number
      • value: number

      Returns this

    • Sets this vector's x, y and z components from the r, g, and b components of the specified Color | color.

      -

      Parameters

      • color: Color

      Returns this

    • Parameters

      • s: Cylindrical

      Returns this

    • Parameters

      • radius: number
      • theta: number
      • y: number

      Returns this

    • Sets this vector's x, y and z components from the x, y, and z components of the specified Euler Euler Angle.

      -

      Parameters

      • e: Euler

      Returns this

    • Parameters

      • matrix: Matrix3
      • index: number

      Returns this

    • Parameters

      • matrix: Matrix4
      • index: number

      Returns this

    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • s: Spherical

      Returns this

    • Parameters

      • r: number
      • phi: number
      • theta: number

      Returns this

    • Normalizes this vector and multiplies it by l.

      -

      Parameters

      • l: number

      Returns this

    • Sets all values of this vector.

      -

      Parameters

      • scalar: number

      Returns this

    • Sets x value of this vector.

      -

      Parameters

      • x: number

      Returns Vector3

    • Sets y value of this vector.

      -

      Parameters

      • y: number

      Returns Vector3

    • Sets z value of this vector.

      -

      Parameters

      • z: number

      Returns Vector3

    • Subtracts v from this vector.

      -

      Parameters

      Returns this

    • Parameters

      • s: number

      Returns this

    • Sets this vector to a - b.

      -

      Parameters

      Returns this

    • Returns an array [x, y, z], or copies x, y and z into the provided array.

      +

      Parameters

      • color: RGB

      Returns this

    • Parameters

      • s: Cylindrical

      Returns this

    • Parameters

      • radius: number
      • theta: number
      • y: number

      Returns this

    • Sets this vector's x, y and z components from the x, y, and z components of the specified Euler Euler Angle.

      +

      Parameters

      • e: Euler

      Returns this

    • Parameters

      • matrix: Matrix3
      • index: number

      Returns this

    • Parameters

      • matrix: Matrix4
      • index: number

      Returns this

    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • s: Spherical

      Returns this

    • Parameters

      • r: number
      • phi: number
      • theta: number

      Returns this

    • Normalizes this vector and multiplies it by l.

      +

      Parameters

      • l: number

      Returns this

    • Sets all values of this vector.

      +

      Parameters

      • scalar: number

      Returns this

    • Sets x value of this vector.

      +

      Parameters

      • x: number

      Returns this

    • Sets y value of this vector.

      +

      Parameters

      • y: number

      Returns this

    • Sets z value of this vector.

      +

      Parameters

      • z: number

      Returns this

    • Subtracts v from this vector.

      +

      Parameters

      • a: Vector3Like

      Returns this

    • Parameters

      • s: number

      Returns this

    • Sets this vector to a - b.

      +

      Parameters

      • a: Vector3Like
      • b: Vector3Like

      Returns this

    • Returns an array [x, y, z], or copies x, y and z into the provided array.

      Parameters

      • Optional array: number[]

        (optional) array to store the vector to. If this is not provided, a new array will be created.

      • Optional offset: number

        (optional) optional offset into the array.

      Returns number[]

      The created or provided array.

      -
    • Parameters

      • Optional array: Vector3Tuple
      • Optional offset: 0

      Returns Vector3Tuple

    • Copies x, y and z into the provided array-like.

      +
    • Parameters

      • Optional array: Vector3Tuple
      • Optional offset: 0

      Returns Vector3Tuple

    • Copies x, y and z into the provided array-like.

      Parameters

      • array: ArrayLike<number>

        array-like to store the vector to.

      • Optional offset: number

        (optional) optional offset into the array-like.

      Returns ArrayLike<number>

      The provided array-like.

      -
    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • camera: Camera

      Returns this

    Generated using TypeDoc

    \ No newline at end of file +
    • Parameters

      • m: Matrix4

      Returns this

    • Parameters

      • camera: Camera

      Returns this

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/classes/VelocityVerletSim.html b/docs/classes/VelocityVerletSim.html index 160d7d9..d2c2dd4 100644 --- a/docs/classes/VelocityVerletSim.html +++ b/docs/classes/VelocityVerletSim.html @@ -1,12 +1,12 @@ VelocityVerletSim | nbody

    Class VelocityVerletSim

    Represents a simulation function object that uses the Velocity Verlet integration method to simulate the motion of bodies.

    -

    Implements

    Constructors

    Implements

    Constructors

    Properties

    Methods

    Constructors

    Properties

    forceCalculator: Force

    Force object to calculate forces on bodies in the Universe.

    -

    Methods

    • Simulate a step in the Universe by using the previous and/or current state and a time step, using the Velocity Verlet integration method.

      +

    Returns VelocityVerletSim

    Properties

    forceCalculator: Force

    Force object to calculate forces on bodies in the Universe.

    +

    Methods

    • Simulate a step in the Universe by using the previous and/or current state and a time step, using the Velocity Verlet integration method.

      Parameters

      • deltaT: number

        time step.

      • currState: State

        current state.

      Returns State

      new state after the simulation step.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/hierarchy.html b/docs/hierarchy.html index f0b150f..502d9dd 100644 --- a/docs/hierarchy.html +++ b/docs/hierarchy.html @@ -1 +1 @@ -nbody

    Generated using TypeDoc

    \ No newline at end of file +nbody

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/interfaces/Force.html b/docs/interfaces/Force.html index d2147a1..c97de59 100644 --- a/docs/interfaces/Force.html +++ b/docs/interfaces/Force.html @@ -1,3 +1,3 @@ Force | nbody

    Interface Force

    Represents a force object used to calculate forces acting on the bodies in the Universe.

    -
    interface Force {
        getForces(bodies): Vector3[];
    }

    Implemented by

    Methods

    Methods

    Generated using TypeDoc

    \ No newline at end of file +
    interface Force {
        getForces(bodies): Vector3[];
    }

    Implemented by

    Methods

    Methods

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/interfaces/SimulateFunction.html b/docs/interfaces/SimulateFunction.html index 9c22c99..e893ad9 100644 --- a/docs/interfaces/SimulateFunction.html +++ b/docs/interfaces/SimulateFunction.html @@ -1,8 +1,8 @@ SimulateFunction | nbody

    Interface SimulateFunction

    Represents a function object used for simulating the Universe. Should encapsulate the numerical integration method and other necessary simulation logic. Can use an external force calculation function object - see Force.

    -
    interface SimulateFunction {
        simulate(deltaT, currState, prevState): State;
    }

    Implemented by

    Methods

    interface SimulateFunction {
        simulate(deltaT, currState, prevState): State;
    }

    Implemented by

    Methods

    Methods

    • Simulate a step in the Universe by using the previous and/or current state and a time step.

      Parameters

      • deltaT: number

        time step.

      • currState: State

        current state of the Universe.

      • prevState: State

        previous state of the Universe.

      Returns State

      the next state of the Universe.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/interfaces/Transformation.html b/docs/interfaces/Transformation.html index 05a3a26..0c9f075 100644 --- a/docs/interfaces/Transformation.html +++ b/docs/interfaces/Transformation.html @@ -1,7 +1,7 @@ Transformation | nbody

    Interface Transformation

    Represents a Frame of Reference transformation.

    -
    interface Transformation {
        transform(state, deltaT): State;
    }

    Implemented by

    Methods

    interface Transformation {
        transform(state, deltaT): State;
    }

    Implemented by

    Methods

    Methods

    • Transform the state to a new frame of reference.

      Parameters

      • state: State

        state to transform.

      • deltaT: number

        time step taken to get to this state. Only applicable for time-dependent transformations.

      Returns State

      transformed state.

      -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/modules.html b/docs/modules.html index c1b3a24..2c8467f 100644 --- a/docs/modules.html +++ b/docs/modules.html @@ -9,6 +9,7 @@

    Forces

    SimulateFunctions

    ExplicitEulerSim LambdaSim diff --git a/docs/types/ControllerType.html b/docs/types/ControllerType.html index 2009b50..bb7861a 100644 --- a/docs/types/ControllerType.html +++ b/docs/types/ControllerType.html @@ -4,4 +4,4 @@
  • 'code' for manual control via code.
  • 'none' for no control.
  • -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/types/UniverseConfig.html b/docs/types/UniverseConfig.html index f88d808..5bd3fec 100644 --- a/docs/types/UniverseConfig.html +++ b/docs/types/UniverseConfig.html @@ -5,4 +5,4 @@
  • prevState: State

    Previous state of the Universe.

  • simFunc: SimulateFunction

    Simulation function used to simulate the Universe.

  • transformations: Transformation[]

    Array of transformations to be applied to the Universe's state after simulation and before visualization.

    -
  • Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/docs/types/VisType.html b/docs/types/VisType.html index 3fdb648..aa8e69b 100644 --- a/docs/types/VisType.html +++ b/docs/types/VisType.html @@ -1,2 +1,2 @@ VisType | nbody

    Type alias VisType

    VisType: "2D" | "3D"

    Visualization type.

    -

    Generated using TypeDoc

    \ No newline at end of file +

    Generated using TypeDoc

    \ No newline at end of file diff --git a/package.json b/package.json index 3cb5550..70357d1 100644 --- a/package.json +++ b/package.json @@ -1,10 +1,10 @@ { "name": "nbody", - "version": "0.0.2", + "version": "0.0.3", "description": "Configure, simulate and visualize n-body interactions", - "main": "./dist/index.js", - "types": "./dist/src/index.d.ts", - "module": "./dist/index.js", + "main": "./dist/src/index.js", + "types": "./dist/types/src/index.d.ts", + "module": "./dist/src/index.js", "repository": "https://github.com/source-academy/nbody.git", "author": "Yeluri Ketan", "license": "Apache-2.0", @@ -29,8 +29,8 @@ "docs": "typedoc", "clean": "node scripts/clean.cjs", "test": "echo \"Error: no test specified\" && exit 1", - "build": "yarn lint && yarn clean && webpack && yarn docs", - "dev": "yarn clean && webpack && yarn unlink && yarn link" + "build": "yarn lint && yarn clean && tsc && yarn docs", + "dev": "yarn clean && tsc && yarn unlink && yarn link" }, "dependencies": { "lil-gui": "^0.19.2" @@ -39,7 +39,7 @@ "@types/eslint": "^8.4.10", "@types/plotly.js": "^2.29.2", "@types/plotly.js-dist": "npm:@types/plotly.js", - "@types/three": "^0.158.1", + "@types/three": "^0.163.0", "@typescript-eslint/eslint-plugin": "^6.6.0", "@typescript-eslint/parser": "^6.6.0", "eslint": "^8.57.0", @@ -54,17 +54,14 @@ "eslint-plugin-react-hooks": "^4.4.0", "eslint-plugin-simple-import-sort": "^8.0.0", "plotly.js-dist": "^2.30.1", - "three": "^0.158.0", - "ts-loader": "^9.5.1", + "three": "^0.163.0", "typedoc": "^0.25.12", - "typescript": "^5.4.2", - "webpack": "^5.90.3", - "webpack-cli": "^5.1.4" + "typescript": "^5.4.2" }, "peerDependencies": { "@types/plotly.js": "^2.29.2", - "@types/three": "^0.158.1", + "@types/three": "^0.163.0", "plotly.js-dist": "^2.30.1", - "three": "^0.158.0" + "three": "^0.163.0" } } \ No newline at end of file diff --git a/src/index.ts b/src/index.ts index 6b42ddb..7af5167 100644 --- a/src/index.ts +++ b/src/index.ts @@ -6,7 +6,7 @@ import { CelestialBody, } from './CelestialBody'; -import { type Force } from './Force'; +import { LambdaForce, type Force } from './Force'; import { CentripetalForce, CombinedForce, Gravity, } from './library/Force'; @@ -38,7 +38,7 @@ import { Vector3 } from 'three'; export { BodyCenterTransformation, CelestialBody, CentripetalForce, - CombinedForce, CoMTransformation, ExplicitEulerSim, Gravity, LambdaSim, LambdaTransformation, RealTimeVisualizer, + CombinedForce, CoMTransformation, ExplicitEulerSim, Gravity, LambdaForce, LambdaSim, LambdaTransformation, RealTimeVisualizer, RealTimeVisualizer3D, RecordingVisualizer, RecordingVisualizer3D, diff --git a/src/library/Visualizer.ts b/src/library/Visualizer.ts index 1ec1164..7ef9cd1 100644 --- a/src/library/Visualizer.ts +++ b/src/library/Visualizer.ts @@ -186,6 +186,8 @@ export class RealTimeVisualizer implements Visualizer { }, // uirevision: 'true', showlegend: false, + width, + height, }; if (this.simulation.controller === 'ui') { @@ -809,6 +811,8 @@ export class RecordingVisualizer implements Visualizer { }, // uirevision: 'true', showlegend: false, + width, + height, }; if (this.simulation.controller === 'ui') { diff --git a/tsconfig.json b/tsconfig.json index 3c07d77..27b41b1 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -3,6 +3,7 @@ "allowJs": true, "allowSyntheticDefaultImports": true, "declaration": true, + "declarationDir": "./dist/types/", "esModuleInterop": true, "forceConsistentCasingInFileNames": true, "isolatedModules": true, @@ -27,5 +28,6 @@ "src/.eslintrc.cjs", ".eslintrc.base.cjs", ".eslintrc.test.cjs", + "node_modules", ] } diff --git a/webpack.config.cjs b/webpack.config.cjs deleted file mode 100644 index 9607043..0000000 --- a/webpack.config.cjs +++ /dev/null @@ -1,41 +0,0 @@ -const path = require("path"); - -module.exports = { - entry: "./src/index.ts", - devtool: "inline-source-map", - mode: "production", - module: { - rules: [ - { - test: /\.tsx?$/, - exclude: /node_modules/, - loader: "ts-loader", - options: { - configFile: path.resolve(__dirname, "tsconfig.json"), - }, - }, - ], - }, - resolve: { - extensions: [".tsx", ".ts", ".js"], - }, - output: { - filename: "index.js", - path: path.resolve(__dirname, "dist"), - globalObject: "this", - library: { - name: "nbody", - type: "umd", - }, - }, - optimization: { - usedExports: false, - }, - externals: { - three: "three", - "three/examples/jsm/Addons": "three/examples/jsm/Addons", - "three/examples/jsm/libs/stats.module": - "three/examples/jsm/libs/stats.module", - "plotly.js-dist": "plotly.js-dist", - }, -}; diff --git a/yarn.lock b/yarn.lock index bb08e54..3d2d48b 100644 --- a/yarn.lock +++ b/yarn.lock @@ -14,11 +14,6 @@ dependencies: regenerator-runtime "^0.14.0" -"@discoveryjs/json-ext@^0.5.0": - version "0.5.7" - resolved "https://registry.yarnpkg.com/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz#1d572bfbbe14b7704e0ba0f39b74815b84870d70" - integrity sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw== - "@es-joy/jsdoccomment@~0.42.0": version "0.42.0" resolved "https://registry.yarnpkg.com/@es-joy/jsdoccomment/-/jsdoccomment-0.42.0.tgz#59e878708336aaee88c2b34c894f73dbf77ae2b0" @@ -79,46 +74,6 @@ resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-2.0.2.tgz#d9fae00a2d5cb40f92cfe64b47ad749fbc38f917" integrity sha512-6EwiSjwWYP7pTckG6I5eyFANjPhmPjUX9JRLUSfNPC7FX7zK9gyZAfUEaECL6ALTpGX5AjnBq3C9XmVWPitNpw== -"@jridgewell/gen-mapping@^0.3.5": - version "0.3.5" - resolved "https://registry.yarnpkg.com/@jridgewell/gen-mapping/-/gen-mapping-0.3.5.tgz#dcce6aff74bdf6dad1a95802b69b04a2fcb1fb36" - integrity sha512-IzL8ZoEDIBRWEzlCcRhOaCupYyN5gdIK+Q6fbFdPDg6HqX6jpkItn7DFIpW9LQzXG6Df9sA7+OKnq0qlz/GaQg== - dependencies: - "@jridgewell/set-array" "^1.2.1" - "@jridgewell/sourcemap-codec" "^1.4.10" - "@jridgewell/trace-mapping" "^0.3.24" - -"@jridgewell/resolve-uri@^3.1.0": - version "3.1.2" - resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz#7a0ee601f60f99a20c7c7c5ff0c80388c1189bd6" - integrity sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw== - -"@jridgewell/set-array@^1.2.1": - version "1.2.1" - resolved "https://registry.yarnpkg.com/@jridgewell/set-array/-/set-array-1.2.1.tgz#558fb6472ed16a4c850b889530e6b36438c49280" - integrity sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A== - -"@jridgewell/source-map@^0.3.3": - version "0.3.6" - resolved "https://registry.yarnpkg.com/@jridgewell/source-map/-/source-map-0.3.6.tgz#9d71ca886e32502eb9362c9a74a46787c36df81a" - integrity sha512-1ZJTZebgqllO79ue2bm3rIGud/bOe0pP5BjSRCRxxYkEZS8STV7zN84UBbiYu7jy+eCKSnVIUgoWWE/tt+shMQ== - dependencies: - "@jridgewell/gen-mapping" "^0.3.5" - "@jridgewell/trace-mapping" "^0.3.25" - -"@jridgewell/sourcemap-codec@^1.4.10", "@jridgewell/sourcemap-codec@^1.4.14": - version "1.4.15" - resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz#d7c6e6755c78567a951e04ab52ef0fd26de59f32" - integrity sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg== - -"@jridgewell/trace-mapping@^0.3.20", "@jridgewell/trace-mapping@^0.3.24", "@jridgewell/trace-mapping@^0.3.25": - version "0.3.25" - resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz#15f190e98895f3fc23276ee14bc76b675c2e50f0" - integrity sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ== - dependencies: - "@jridgewell/resolve-uri" "^3.1.0" - "@jridgewell/sourcemap-codec" "^1.4.14" - "@nodelib/fs.scandir@2.1.5": version "2.1.5" resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" @@ -140,15 +95,12 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" -"@types/eslint-scope@^3.7.3": - version "3.7.7" - resolved "https://registry.yarnpkg.com/@types/eslint-scope/-/eslint-scope-3.7.7.tgz#3108bd5f18b0cdb277c867b3dd449c9ed7079ac5" - integrity sha512-MzMFlSLBqNF2gcHWO0G1vP/YQyfvrxZ0bF+u7mzUdZ1/xK4A4sru+nraZz5i3iEIk1l1uyicaDVTB4QbbEkAYg== - dependencies: - "@types/eslint" "*" - "@types/estree" "*" +"@tweenjs/tween.js@~23.1.1": + version "23.1.1" + resolved "https://registry.yarnpkg.com/@tweenjs/tween.js/-/tween.js-23.1.1.tgz#0ae28ed9c635805557f78c2626464018d5f1b5e2" + integrity sha512-ZpboH7pCPPeyBWKf8c7TJswtCEQObFo3bOBYalm99NzZarATALYCo5OhbCa/n4RQyJyHfhkdx+hNrdL5ByFYDw== -"@types/eslint@*", "@types/eslint@^8.4.10": +"@types/eslint@^8.4.10": version "8.56.5" resolved "https://registry.yarnpkg.com/@types/eslint/-/eslint-8.56.5.tgz#94b88cab77588fcecdd0771a6d576fa1c0af9d02" integrity sha512-u5/YPJHo1tvkSF2CE0USEkxon82Z5DBy2xR+qfyYNszpX9qcs4sT6uq2kBbj4BXY1+DBGDPnrhMZV3pKWGNukw== @@ -156,12 +108,12 @@ "@types/estree" "*" "@types/json-schema" "*" -"@types/estree@*", "@types/estree@^1.0.5": +"@types/estree@*": version "1.0.5" resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" integrity sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw== -"@types/json-schema@*", "@types/json-schema@^7.0.12", "@types/json-schema@^7.0.8", "@types/json-schema@^7.0.9": +"@types/json-schema@*", "@types/json-schema@^7.0.12", "@types/json-schema@^7.0.9": version "7.0.15" resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== @@ -171,13 +123,6 @@ resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee" integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ== -"@types/node@*": - version "20.11.28" - resolved "https://registry.yarnpkg.com/@types/node/-/node-20.11.28.tgz#4fd5b2daff2e580c12316e457473d68f15ee6f66" - integrity sha512-M/GPWVS2wLkSkNHVeLkrF2fD5Lx5UC4PxA0uZcKc6QqbIQUJyW1jVjueJYi1z8n0I5PxYrtpnPnWglE+y9A0KA== - dependencies: - undici-types "~5.26.4" - "@types/plotly.js-dist@npm:@types/plotly.js", "@types/plotly.js@^2.29.2": version "2.29.2" resolved "https://registry.yarnpkg.com/@types/plotly.js/-/plotly.js-2.29.2.tgz#e058cd17963c1658c2bd204b2b9b1b782cdf73d7" @@ -193,14 +138,15 @@ resolved "https://registry.yarnpkg.com/@types/stats.js/-/stats.js-0.17.3.tgz#705446e12ce0fad618557dd88236f51148b7a935" integrity sha512-pXNfAD3KHOdif9EQXZ9deK82HVNaXP5ZIF5RP2QG6OQFNTaY2YIetfrE9t528vEreGQvEPRDDc8muaoYeK0SxQ== -"@types/three@^0.158.1": - version "0.158.3" - resolved "https://registry.yarnpkg.com/@types/three/-/three-0.158.3.tgz#f53e26c50709c7f972934e4e8c1ca51add0cfa9f" - integrity sha512-6Qs1rUvLSbkJ4hlIe6/rdwIf61j1x2UKvGJg7s8KjswYsz1C1qDTs6voVXXB8kYaI0hgklgZgbZUupfL1l9xdA== +"@types/three@^0.163.0": + version "0.163.0" + resolved "https://registry.yarnpkg.com/@types/three/-/three-0.163.0.tgz#96f5440fcd39452d2c84dfe0c9b7a9cf0247b9e6" + integrity sha512-uIdDhsXRpQiBUkflBS/i1l3JX14fW6Ot9csed60nfbZNXHDTRsnV2xnTVwXcgbvTiboAR4IW+t+lTL5f1rqIqA== dependencies: + "@tweenjs/tween.js" "~23.1.1" "@types/stats.js" "*" "@types/webxr" "*" - fflate "~0.6.10" + fflate "~0.8.2" meshoptimizer "~0.18.1" "@types/webxr@*": @@ -347,173 +293,17 @@ resolved "https://registry.yarnpkg.com/@ungap/structured-clone/-/structured-clone-1.2.0.tgz#756641adb587851b5ccb3e095daf27ae581c8406" integrity sha512-zuVdFrMJiuCDQUMCzQaD6KL28MjnqqN8XnAqiEq9PNm/hCPTSGfrXCOfwj1ow4LFb/tNymJPwsNbVePc1xFqrQ== -"@webassemblyjs/ast@1.12.1", "@webassemblyjs/ast@^1.11.5": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/ast/-/ast-1.12.1.tgz#bb16a0e8b1914f979f45864c23819cc3e3f0d4bb" - integrity sha512-EKfMUOPRRUTy5UII4qJDGPpqfwjOmZ5jeGFwid9mnoqIFK+e0vqoi1qH56JpmZSzEL53jKnNzScdmftJyG5xWg== - dependencies: - "@webassemblyjs/helper-numbers" "1.11.6" - "@webassemblyjs/helper-wasm-bytecode" "1.11.6" - -"@webassemblyjs/floating-point-hex-parser@1.11.6": - version "1.11.6" - resolved "https://registry.yarnpkg.com/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.11.6.tgz#dacbcb95aff135c8260f77fa3b4c5fea600a6431" - integrity sha512-ejAj9hfRJ2XMsNHk/v6Fu2dGS+i4UaXBXGemOfQ/JfQ6mdQg/WXtwleQRLLS4OvfDhv8rYnVwH27YJLMyYsxhw== - -"@webassemblyjs/helper-api-error@1.11.6": - version "1.11.6" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-api-error/-/helper-api-error-1.11.6.tgz#6132f68c4acd59dcd141c44b18cbebbd9f2fa768" - integrity sha512-o0YkoP4pVu4rN8aTJgAyj9hC2Sv5UlkzCHhxqWj8butaLvnpdc2jOwh4ewE6CX0txSfLn/UYaV/pheS2Txg//Q== - -"@webassemblyjs/helper-buffer@1.12.1": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-buffer/-/helper-buffer-1.12.1.tgz#6df20d272ea5439bf20ab3492b7fb70e9bfcb3f6" - integrity sha512-nzJwQw99DNDKr9BVCOZcLuJJUlqkJh+kVzVl6Fmq/tI5ZtEyWT1KZMyOXltXLZJmDtvLCDgwsyrkohEtopTXCw== - -"@webassemblyjs/helper-numbers@1.11.6": - version "1.11.6" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-numbers/-/helper-numbers-1.11.6.tgz#cbce5e7e0c1bd32cf4905ae444ef64cea919f1b5" - integrity sha512-vUIhZ8LZoIWHBohiEObxVm6hwP034jwmc9kuq5GdHZH0wiLVLIPcMCdpJzG4C11cHoQ25TFIQj9kaVADVX7N3g== - dependencies: - "@webassemblyjs/floating-point-hex-parser" "1.11.6" - "@webassemblyjs/helper-api-error" "1.11.6" - "@xtuc/long" "4.2.2" - -"@webassemblyjs/helper-wasm-bytecode@1.11.6": - version "1.11.6" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.11.6.tgz#bb2ebdb3b83aa26d9baad4c46d4315283acd51e9" - integrity sha512-sFFHKwcmBprO9e7Icf0+gddyWYDViL8bpPjJJl0WHxCdETktXdmtWLGVzoHbqUcY4Be1LkNfwTmXOJUFZYSJdA== - -"@webassemblyjs/helper-wasm-section@1.12.1": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.12.1.tgz#3da623233ae1a60409b509a52ade9bc22a37f7bf" - integrity sha512-Jif4vfB6FJlUlSbgEMHUyk1j234GTNG9dBJ4XJdOySoj518Xj0oGsNi59cUQF4RRMS9ouBUxDDdyBVfPTypa5g== - dependencies: - "@webassemblyjs/ast" "1.12.1" - "@webassemblyjs/helper-buffer" "1.12.1" - "@webassemblyjs/helper-wasm-bytecode" "1.11.6" - "@webassemblyjs/wasm-gen" "1.12.1" - -"@webassemblyjs/ieee754@1.11.6": - version "1.11.6" - resolved "https://registry.yarnpkg.com/@webassemblyjs/ieee754/-/ieee754-1.11.6.tgz#bb665c91d0b14fffceb0e38298c329af043c6e3a" - integrity sha512-LM4p2csPNvbij6U1f19v6WR56QZ8JcHg3QIJTlSwzFcmx6WSORicYj6I63f9yU1kEUtrpG+kjkiIAkevHpDXrg== - dependencies: - "@xtuc/ieee754" "^1.2.0" - -"@webassemblyjs/leb128@1.11.6": - version "1.11.6" - resolved "https://registry.yarnpkg.com/@webassemblyjs/leb128/-/leb128-1.11.6.tgz#70e60e5e82f9ac81118bc25381a0b283893240d7" - integrity sha512-m7a0FhE67DQXgouf1tbN5XQcdWoNgaAuoULHIfGFIEVKA6tu/edls6XnIlkmS6FrXAquJRPni3ZZKjw6FSPjPQ== - dependencies: - "@xtuc/long" "4.2.2" - -"@webassemblyjs/utf8@1.11.6": - version "1.11.6" - resolved "https://registry.yarnpkg.com/@webassemblyjs/utf8/-/utf8-1.11.6.tgz#90f8bc34c561595fe156603be7253cdbcd0fab5a" - integrity sha512-vtXf2wTQ3+up9Zsg8sa2yWiQpzSsMyXj0qViVP6xKGCUT8p8YJ6HqI7l5eCnWx1T/FYdsv07HQs2wTFbbof/RA== - -"@webassemblyjs/wasm-edit@^1.11.5": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-edit/-/wasm-edit-1.12.1.tgz#9f9f3ff52a14c980939be0ef9d5df9ebc678ae3b" - integrity sha512-1DuwbVvADvS5mGnXbE+c9NfA8QRcZ6iKquqjjmR10k6o+zzsRVesil54DKexiowcFCPdr/Q0qaMgB01+SQ1u6g== - dependencies: - "@webassemblyjs/ast" "1.12.1" - "@webassemblyjs/helper-buffer" "1.12.1" - "@webassemblyjs/helper-wasm-bytecode" "1.11.6" - "@webassemblyjs/helper-wasm-section" "1.12.1" - "@webassemblyjs/wasm-gen" "1.12.1" - "@webassemblyjs/wasm-opt" "1.12.1" - "@webassemblyjs/wasm-parser" "1.12.1" - "@webassemblyjs/wast-printer" "1.12.1" - -"@webassemblyjs/wasm-gen@1.12.1": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-gen/-/wasm-gen-1.12.1.tgz#a6520601da1b5700448273666a71ad0a45d78547" - integrity sha512-TDq4Ojh9fcohAw6OIMXqiIcTq5KUXTGRkVxbSo1hQnSy6lAM5GSdfwWeSxpAo0YzgsgF182E/U0mDNhuA0tW7w== - dependencies: - "@webassemblyjs/ast" "1.12.1" - "@webassemblyjs/helper-wasm-bytecode" "1.11.6" - "@webassemblyjs/ieee754" "1.11.6" - "@webassemblyjs/leb128" "1.11.6" - "@webassemblyjs/utf8" "1.11.6" - -"@webassemblyjs/wasm-opt@1.12.1": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-opt/-/wasm-opt-1.12.1.tgz#9e6e81475dfcfb62dab574ac2dda38226c232bc5" - integrity sha512-Jg99j/2gG2iaz3hijw857AVYekZe2SAskcqlWIZXjji5WStnOpVoat3gQfT/Q5tb2djnCjBtMocY/Su1GfxPBg== - dependencies: - "@webassemblyjs/ast" "1.12.1" - "@webassemblyjs/helper-buffer" "1.12.1" - "@webassemblyjs/wasm-gen" "1.12.1" - "@webassemblyjs/wasm-parser" "1.12.1" - -"@webassemblyjs/wasm-parser@1.12.1", "@webassemblyjs/wasm-parser@^1.11.5": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wasm-parser/-/wasm-parser-1.12.1.tgz#c47acb90e6f083391e3fa61d113650eea1e95937" - integrity sha512-xikIi7c2FHXysxXe3COrVUPSheuBtpcfhbpFj4gmu7KRLYOzANztwUU0IbsqvMqzuNK2+glRGWCEqZo1WCLyAQ== - dependencies: - "@webassemblyjs/ast" "1.12.1" - "@webassemblyjs/helper-api-error" "1.11.6" - "@webassemblyjs/helper-wasm-bytecode" "1.11.6" - "@webassemblyjs/ieee754" "1.11.6" - "@webassemblyjs/leb128" "1.11.6" - "@webassemblyjs/utf8" "1.11.6" - -"@webassemblyjs/wast-printer@1.12.1": - version "1.12.1" - resolved "https://registry.yarnpkg.com/@webassemblyjs/wast-printer/-/wast-printer-1.12.1.tgz#bcecf661d7d1abdaf989d8341a4833e33e2b31ac" - integrity sha512-+X4WAlOisVWQMikjbcvY2e0rwPsKQ9F688lksZhBcPycBBuii3O7m8FACbDMWDojpAqvjIncrG8J0XHKyQfVeA== - dependencies: - "@webassemblyjs/ast" "1.12.1" - "@xtuc/long" "4.2.2" - -"@webpack-cli/configtest@^2.1.1": - version "2.1.1" - resolved "https://registry.yarnpkg.com/@webpack-cli/configtest/-/configtest-2.1.1.tgz#3b2f852e91dac6e3b85fb2a314fb8bef46d94646" - integrity sha512-wy0mglZpDSiSS0XHrVR+BAdId2+yxPSoJW8fsna3ZpYSlufjvxnP4YbKTCBZnNIcGN4r6ZPXV55X4mYExOfLmw== - -"@webpack-cli/info@^2.0.2": - version "2.0.2" - resolved "https://registry.yarnpkg.com/@webpack-cli/info/-/info-2.0.2.tgz#cc3fbf22efeb88ff62310cf885c5b09f44ae0fdd" - integrity sha512-zLHQdI/Qs1UyT5UBdWNqsARasIA+AaF8t+4u2aS2nEpBQh2mWIVb8qAklq0eUENnC5mOItrIB4LiS9xMtph18A== - -"@webpack-cli/serve@^2.0.5": - version "2.0.5" - resolved "https://registry.yarnpkg.com/@webpack-cli/serve/-/serve-2.0.5.tgz#325db42395cd49fe6c14057f9a900e427df8810e" - integrity sha512-lqaoKnRYBdo1UgDX8uF24AfGMifWK19TxPmM5FHc2vAGxrJ/qtyUyFBWoY1tISZdelsQ5fBcOusifo5o5wSJxQ== - -"@xtuc/ieee754@^1.2.0": - version "1.2.0" - resolved "https://registry.yarnpkg.com/@xtuc/ieee754/-/ieee754-1.2.0.tgz#eef014a3145ae477a1cbc00cd1e552336dceb790" - integrity sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA== - -"@xtuc/long@4.2.2": - version "4.2.2" - resolved "https://registry.yarnpkg.com/@xtuc/long/-/long-4.2.2.tgz#d291c6a4e97989b5c61d9acf396ae4fe133a718d" - integrity sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ== - -acorn-import-assertions@^1.9.0: - version "1.9.0" - resolved "https://registry.yarnpkg.com/acorn-import-assertions/-/acorn-import-assertions-1.9.0.tgz#507276249d684797c84e0734ef84860334cfb1ac" - integrity sha512-cmMwop9x+8KFhxvKrKfPYmN6/pKTYYHBqLa0DfvVZcKMJWNyWLnaqND7dx/qn66R7ewM1UX5XMaDVP5wlVTaVA== - acorn-jsx@^5.3.2: version "5.3.2" resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937" integrity sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ== -acorn@^8.7.1, acorn@^8.8.2, acorn@^8.9.0: +acorn@^8.9.0: version "8.11.3" resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.11.3.tgz#71e0b14e13a4ec160724b38fb7b0f233b1b81d7a" integrity sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg== -ajv-keywords@^3.5.2: - version "3.5.2" - resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d" - integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== - -ajv@^6.12.4, ajv@^6.12.5: +ajv@^6.12.4: version "6.12.6" resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== @@ -720,21 +510,6 @@ braces@^3.0.2: dependencies: fill-range "^7.0.1" -browserslist@^4.21.10: - version "4.23.0" - resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.23.0.tgz#8f3acc2bbe73af7213399430890f86c63a5674ab" - integrity sha512-QW8HiM1shhT2GuzkvklfjcKDiWFXHOeFCIA/huJPwHsslwcydgk7X+z2zXpEijP98UCY7HbubZt5J2Zgvf0CaQ== - dependencies: - caniuse-lite "^1.0.30001587" - electron-to-chromium "^1.4.668" - node-releases "^2.0.14" - update-browserslist-db "^1.0.13" - -buffer-from@^1.0.0: - version "1.1.2" - resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.2.tgz#2b146a6fd72e80b4f55d255f35ed59a3a9a41bd5" - integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ== - builtin-modules@^3.3.0: version "3.3.0" resolved "https://registry.yarnpkg.com/builtin-modules/-/builtin-modules-3.3.0.tgz#cae62812b89801e9656336e46223e030386be7b6" @@ -756,12 +531,7 @@ callsites@^3.0.0: resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ== -caniuse-lite@^1.0.30001587: - version "1.0.30001599" - resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001599.tgz#571cf4f3f1506df9bf41fcbb6d10d5d017817bce" - integrity sha512-LRAQHZ4yT1+f9LemSMeqdMpMxZcc4RMWdj4tiFe3G8tNkWK+E58g+/tzotb5cU6TbcVJLr4fySiAW7XmxQvZQA== - -chalk@^4.0.0, chalk@^4.1.0: +chalk@^4.0.0: version "4.1.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== @@ -769,20 +539,6 @@ chalk@^4.0.0, chalk@^4.1.0: ansi-styles "^4.1.0" supports-color "^7.1.0" -chrome-trace-event@^1.0.2: - version "1.0.3" - resolved "https://registry.yarnpkg.com/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz#1015eced4741e15d06664a957dbbf50d041e26ac" - integrity sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg== - -clone-deep@^4.0.1: - version "4.0.1" - resolved "https://registry.yarnpkg.com/clone-deep/-/clone-deep-4.0.1.tgz#c19fd9bdbbf85942b4fd979c84dcf7d5f07c2387" - integrity sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ== - dependencies: - is-plain-object "^2.0.4" - kind-of "^6.0.2" - shallow-clone "^3.0.0" - color-convert@^2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" @@ -795,21 +551,6 @@ color-name@~1.1.4: resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== -colorette@^2.0.14: - version "2.0.20" - resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.20.tgz#9eb793e6833067f7235902fcd3b09917a000a95a" - integrity sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w== - -commander@^10.0.1: - version "10.0.1" - resolved "https://registry.yarnpkg.com/commander/-/commander-10.0.1.tgz#881ee46b4f77d1c1dccc5823433aa39b022cbe06" - integrity sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug== - -commander@^2.20.0: - version "2.20.3" - resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33" - integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ== - comment-parser@1.4.1: version "1.4.1" resolved "https://registry.yarnpkg.com/comment-parser/-/comment-parser-1.4.1.tgz#bdafead37961ac079be11eb7ec65c4d021eaf9cc" @@ -825,7 +566,7 @@ confusing-browser-globals@^1.0.10: resolved "https://registry.yarnpkg.com/confusing-browser-globals/-/confusing-browser-globals-1.0.11.tgz#ae40e9b57cdd3915408a2805ebd3a5585608dc81" integrity sha512-JsPKdmh8ZkmnHxDk55FZ1TqVLvEQTvoByJZRN9jzI0UjxK/QgAmsphz7PGtqgPieQZ/CQcHWXCR7ATDNhGe+YA== -cross-spawn@^7.0.2, cross-spawn@^7.0.3: +cross-spawn@^7.0.2: version "7.0.3" resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.3.tgz#f73a85b9d5d41d045551c177e2882d4ac85728a6" integrity sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w== @@ -929,29 +670,11 @@ doctrine@^3.0.0: dependencies: esutils "^2.0.2" -electron-to-chromium@^1.4.668: - version "1.4.708" - resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.708.tgz#d54d3b47cb44ae6b190067439c42135456907893" - integrity sha512-iWgEEvREL4GTXXHKohhh33+6Y8XkPI5eHihDmm8zUk5Zo7HICEW+wI/j5kJ2tbuNUCXJ/sNXa03ajW635DiJXA== - emoji-regex@^9.2.2: version "9.2.2" resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.2.2.tgz#840c8803b0d8047f4ff0cf963176b32d4ef3ed72" integrity sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg== -enhanced-resolve@^5.0.0, enhanced-resolve@^5.15.0: - version "5.16.0" - resolved "https://registry.yarnpkg.com/enhanced-resolve/-/enhanced-resolve-5.16.0.tgz#65ec88778083056cb32487faa9aef82ed0864787" - integrity sha512-O+QWCviPNSSLAD9Ucn8Awv+poAkqn3T1XY5/N7kR7rQO9yfSGWkYZDwpJ+iKF7B8rxaQKWngSqACpgzeapSyoA== - dependencies: - graceful-fs "^4.2.4" - tapable "^2.2.0" - -envinfo@^7.7.3: - version "7.11.1" - resolved "https://registry.yarnpkg.com/envinfo/-/envinfo-7.11.1.tgz#2ffef77591057081b0129a8fd8cf6118da1b94e1" - integrity sha512-8PiZgZNIB4q/Lw4AhOvAfB/ityHAd2bli3lESSWmWSzSsl5dKpy5N1d1Rfkd2teq/g9xN90lc6o98DOjMeYHpg== - es-abstract@^1.22.1, es-abstract@^1.22.3: version "1.22.5" resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.22.5.tgz#1417df4e97cc55f09bf7e58d1e614bc61cb8df46" @@ -1088,11 +811,6 @@ es-iterator-helpers@^1.0.15, es-iterator-helpers@^1.0.17: iterator.prototype "^1.1.2" safe-array-concat "^1.1.2" -es-module-lexer@^1.2.1: - version "1.4.1" - resolved "https://registry.yarnpkg.com/es-module-lexer/-/es-module-lexer-1.4.1.tgz#41ea21b43908fe6a287ffcbe4300f790555331f5" - integrity sha512-cXLGjP0c4T3flZJKQSuziYoq7MlT+rnvfZjfp7h+I7K9BNX54kP9nyWvdbwjQ4u1iWbOL4u96fgeZLToQlZC7w== - es-object-atoms@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/es-object-atoms/-/es-object-atoms-1.0.0.tgz#ddb55cd47ac2e240701260bc2a8e31ecb643d941" @@ -1125,11 +843,6 @@ es-to-primitive@^1.2.1: is-date-object "^1.0.1" is-symbol "^1.0.2" -escalade@^3.1.1: - version "3.1.2" - resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.1.2.tgz#54076e9ab29ea5bf3d8f1ed62acffbb88272df27" - integrity sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA== - escape-string-regexp@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" @@ -1289,7 +1002,7 @@ eslint-plugin-simple-import-sort@^8.0.0: resolved "https://registry.yarnpkg.com/eslint-plugin-simple-import-sort/-/eslint-plugin-simple-import-sort-8.0.0.tgz#9d9a2372b0606e999ea841b10458a370a6ccc160" integrity sha512-bXgJQ+lqhtQBCuWY/FUWdB27j4+lqcvXv5rUARkzbeWLwea+S5eBZEQrhnO+WgX3ZoJHVj0cn943iyXwByHHQw== -eslint-scope@5.1.1, eslint-scope@^5.1.1: +eslint-scope@^5.1.1: version "5.1.1" resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-5.1.1.tgz#e786e59a66cb92b3f6c1fb0d508aab174848f48c" integrity sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw== @@ -1392,11 +1105,6 @@ esutils@^2.0.2: resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64" integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== -events@^3.2.0: - version "3.3.0" - resolved "https://registry.yarnpkg.com/events/-/events-3.3.0.tgz#31a95ad0a924e2d2c419a813aeb2c4e878ea7400" - integrity sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q== - fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: version "3.1.3" resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" @@ -1423,11 +1131,6 @@ fast-levenshtein@^2.0.6: resolved "https://registry.yarnpkg.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917" integrity sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw== -fastest-levenshtein@^1.0.12: - version "1.0.16" - resolved "https://registry.yarnpkg.com/fastest-levenshtein/-/fastest-levenshtein-1.0.16.tgz#210e61b6ff181de91ea9b3d1b84fdedd47e034e5" - integrity sha512-eRnCtTTtGZFpQCwhJiUOuxPQWRXVKYDn0b2PeHfXL6/Zi53SLAzAHfVhVWK2AryC/WH05kGfxhFIPvTF0SXQzg== - fastq@^1.6.0: version "1.17.1" resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.17.1.tgz#2a523f07a4e7b1e81a42b91b8bf2254107753b47" @@ -1435,10 +1138,10 @@ fastq@^1.6.0: dependencies: reusify "^1.0.4" -fflate@~0.6.10: - version "0.6.10" - resolved "https://registry.yarnpkg.com/fflate/-/fflate-0.6.10.tgz#5f40f9659205936a2d18abf88b2e7781662b6d43" - integrity sha512-IQrh3lEPM93wVCEczc9SaAOvkmcoQn/G8Bo1e8ZPlY3X3bnAxWaBdvTdvM1hP62iZp0BXWDy4vTAy4fF0+Dlpg== +fflate@~0.8.2: + version "0.8.2" + resolved "https://registry.yarnpkg.com/fflate/-/fflate-0.8.2.tgz#fc8631f5347812ad6028bbe4a2308b2792aa1dea" + integrity sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A== file-entry-cache@^6.0.1: version "6.0.1" @@ -1454,14 +1157,6 @@ fill-range@^7.0.1: dependencies: to-regex-range "^5.0.1" -find-up@^4.0.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/find-up/-/find-up-4.1.0.tgz#97afe7d6cdc0bc5928584b7c8d7b16e8a9aa5d19" - integrity sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw== - dependencies: - locate-path "^5.0.0" - path-exists "^4.0.0" - find-up@^5.0.0: version "5.0.0" resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc" @@ -1479,11 +1174,6 @@ flat-cache@^3.0.4: keyv "^4.5.3" rimraf "^3.0.2" -flat@^5.0.2: - version "5.0.2" - resolved "https://registry.yarnpkg.com/flat/-/flat-5.0.2.tgz#8ca6fe332069ffa9d324c327198c598259ceb241" - integrity sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ== - flatted@^3.2.9: version "3.3.1" resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.3.1.tgz#21db470729a6734d4997002f439cb308987f567a" @@ -1555,11 +1245,6 @@ glob-parent@^6.0.2: dependencies: is-glob "^4.0.3" -glob-to-regexp@^0.4.1: - version "0.4.1" - resolved "https://registry.yarnpkg.com/glob-to-regexp/-/glob-to-regexp-0.4.1.tgz#c75297087c851b9a578bd217dd59a92f59fe546e" - integrity sha512-lkX1HJXwyMcprw/5YUZc2s7DrpAiHB21/V+E1rHUrVNokkvB6bqMzT0VfV6/86ZNabt1k14YOIaT7nDvOX3Iiw== - glob@^7.1.3, glob@^7.2.0: version "7.2.3" resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" @@ -1605,11 +1290,6 @@ gopd@^1.0.1: dependencies: get-intrinsic "^1.1.3" -graceful-fs@^4.1.2, graceful-fs@^4.2.4, graceful-fs@^4.2.9: - version "4.2.11" - resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" - integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ== - graphemer@^1.4.0: version "1.4.0" resolved "https://registry.yarnpkg.com/graphemer/-/graphemer-1.4.0.tgz#fb2f1d55e0e3a1849aeffc90c4fa0dd53a0e66c6" @@ -1669,14 +1349,6 @@ import-fresh@^3.2.1: parent-module "^1.0.0" resolve-from "^4.0.0" -import-local@^3.0.2: - version "3.1.0" - resolved "https://registry.yarnpkg.com/import-local/-/import-local-3.1.0.tgz#b4479df8a5fd44f6cdce24070675676063c95cb4" - integrity sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg== - dependencies: - pkg-dir "^4.2.0" - resolve-cwd "^3.0.0" - imurmurhash@^0.1.4: version "0.1.4" resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" @@ -1704,11 +1376,6 @@ internal-slot@^1.0.5, internal-slot@^1.0.7: hasown "^2.0.0" side-channel "^1.0.4" -interpret@^3.1.1: - version "3.1.1" - resolved "https://registry.yarnpkg.com/interpret/-/interpret-3.1.1.tgz#5be0ceed67ca79c6c4bc5cf0d7ee843dcea110c4" - integrity sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ== - is-array-buffer@^3.0.4: version "3.0.4" resolved "https://registry.yarnpkg.com/is-array-buffer/-/is-array-buffer-3.0.4.tgz#7a1f92b3d61edd2bc65d24f130530ea93d7fae98" @@ -1825,13 +1492,6 @@ is-path-inside@^3.0.3: resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz#d231362e53a07ff2b0e0ea7fed049161ffd16283" integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ== -is-plain-object@^2.0.4: - version "2.0.4" - resolved "https://registry.yarnpkg.com/is-plain-object/-/is-plain-object-2.0.4.tgz#2c163b3fafb1b606d9d17928f05c2a1c38e07677" - integrity sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og== - dependencies: - isobject "^3.0.1" - is-regex@^1.1.4: version "1.1.4" resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.1.4.tgz#eef5663cd59fa4c0ae339505323df6854bb15958" @@ -1903,11 +1563,6 @@ isexe@^2.0.0: resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== -isobject@^3.0.1: - version "3.0.1" - resolved "https://registry.yarnpkg.com/isobject/-/isobject-3.0.1.tgz#4e431e92b11a9731636aa1f9c8d1ccbcfdab78df" - integrity sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg== - iterator.prototype@^1.1.2: version "1.1.2" resolved "https://registry.yarnpkg.com/iterator.prototype/-/iterator.prototype-1.1.2.tgz#5e29c8924f01916cb9335f1ff80619dcff22b0c0" @@ -1919,15 +1574,6 @@ iterator.prototype@^1.1.2: reflect.getprototypeof "^1.0.4" set-function-name "^2.0.1" -jest-worker@^27.4.5: - version "27.5.1" - resolved "https://registry.yarnpkg.com/jest-worker/-/jest-worker-27.5.1.tgz#8d146f0900e8973b106b6f73cc1e9a8cb86f8db0" - integrity sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg== - dependencies: - "@types/node" "*" - merge-stream "^2.0.0" - supports-color "^8.0.0" - "js-tokens@^3.0.0 || ^4.0.0": version "4.0.0" resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499" @@ -1950,11 +1596,6 @@ json-buffer@3.0.1: resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.1.tgz#9338802a30d3b6605fbe0613e094008ca8c05a13" integrity sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ== -json-parse-even-better-errors@^2.3.1: - version "2.3.1" - resolved "https://registry.yarnpkg.com/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz#7c47805a94319928e05777405dc12e1f7a4ee02d" - integrity sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w== - json-schema-traverse@^0.4.1: version "0.4.1" resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" @@ -1994,11 +1635,6 @@ keyv@^4.5.3: dependencies: json-buffer "3.0.1" -kind-of@^6.0.2: - version "6.0.3" - resolved "https://registry.yarnpkg.com/kind-of/-/kind-of-6.0.3.tgz#07c05034a6c349fa06e24fa35aa76db4580ce4dd" - integrity sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw== - language-subtag-registry@^0.3.20: version "0.3.22" resolved "https://registry.yarnpkg.com/language-subtag-registry/-/language-subtag-registry-0.3.22.tgz#2e1500861b2e457eba7e7ae86877cbd08fa1fd1d" @@ -2024,18 +1660,6 @@ lil-gui@^0.19.2: resolved "https://registry.yarnpkg.com/lil-gui/-/lil-gui-0.19.2.tgz#774bf46f1f43ec5fec00884af46b8060786f97fc" integrity sha512-nU8j4ND702ouGfQZoaTN4dfXxacvGOAVK0DtmZBVcUYUAeYQXLQAjAN50igMHiba3T5jZyKEjXZU+Ntm1Qs6ZQ== -loader-runner@^4.2.0: - version "4.3.0" - resolved "https://registry.yarnpkg.com/loader-runner/-/loader-runner-4.3.0.tgz#c1b4a163b99f614830353b16755e7149ac2314e1" - integrity sha512-3R/1M+yS3j5ou80Me59j7F9IMs4PXs3VqRrm0TU3AbKPxlmpoY1TNscJV/oGJXo8qCatFGTfDbY6W6ipGOYXfg== - -locate-path@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-5.0.0.tgz#1afba396afd676a6d42504d0a67a3a7eb9f62aa0" - integrity sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g== - dependencies: - p-locate "^4.1.0" - locate-path@^6.0.0: version "6.0.0" resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286" @@ -2072,11 +1696,6 @@ marked@^4.3.0: resolved "https://registry.yarnpkg.com/marked/-/marked-4.3.0.tgz#796362821b019f734054582038b116481b456cf3" integrity sha512-PRsaiG84bK+AMvxziE/lCFss8juXjNaWzVbN5tXAm4XjeaS9NAHhop+PjQxz2A9h8Q4M/xGmzP8vqNwy6JeK0A== -merge-stream@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" - integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== - merge2@^1.3.0, merge2@^1.4.1: version "1.4.1" resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae" @@ -2087,7 +1706,7 @@ meshoptimizer@~0.18.1: resolved "https://registry.yarnpkg.com/meshoptimizer/-/meshoptimizer-0.18.1.tgz#cdb90907f30a7b5b1190facd3b7ee6b7087797d8" integrity sha512-ZhoIoL7TNV4s5B6+rx5mC//fw8/POGyNxS/DZyCJeiZ12ScLfVwRE/GfsxwiTkMYYD5DmK2/JXnEVXqL4rF+Sw== -micromatch@^4.0.0, micromatch@^4.0.4: +micromatch@^4.0.4: version "4.0.5" resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.5.tgz#bc8999a7cbbf77cdc89f132f6e467051b49090c6" integrity sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA== @@ -2095,18 +1714,6 @@ micromatch@^4.0.0, micromatch@^4.0.4: braces "^3.0.2" picomatch "^2.3.1" -mime-db@1.52.0: - version "1.52.0" - resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" - integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== - -mime-types@^2.1.27: - version "2.1.35" - resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" - integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== - dependencies: - mime-db "1.52.0" - minimatch@9.0.3, minimatch@^9.0.3: version "9.0.3" resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.3.tgz#a6e00c3de44c3a542bfaae70abfc22420a6da825" @@ -2141,16 +1748,6 @@ natural-compare@^1.4.0: resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== -neo-async@^2.6.2: - version "2.6.2" - resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.2.tgz#b4aafb93e3aeb2d8174ca53cf163ab7d7308305f" - integrity sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw== - -node-releases@^2.0.14: - version "2.0.14" - resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.14.tgz#2ffb053bceb8b2be8495ece1ab6ce600c4461b0b" - integrity sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw== - object-assign@^4.1.1: version "4.1.1" resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" @@ -2241,13 +1838,6 @@ optionator@^0.9.3: prelude-ls "^1.2.1" type-check "^0.4.0" -p-limit@^2.2.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-2.3.0.tgz#3dd33c647a214fdfffd835933eb086da0dc21db1" - integrity sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w== - dependencies: - p-try "^2.0.0" - p-limit@^3.0.2: version "3.1.0" resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b" @@ -2255,13 +1845,6 @@ p-limit@^3.0.2: dependencies: yocto-queue "^0.1.0" -p-locate@^4.1.0: - version "4.1.0" - resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-4.1.0.tgz#a3428bb7088b3a60292f66919278b7c297ad4f07" - integrity sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A== - dependencies: - p-limit "^2.2.0" - p-locate@^5.0.0: version "5.0.0" resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834" @@ -2269,11 +1852,6 @@ p-locate@^5.0.0: dependencies: p-limit "^3.0.2" -p-try@^2.0.0: - version "2.2.0" - resolved "https://registry.yarnpkg.com/p-try/-/p-try-2.2.0.tgz#cb2868540e313d61de58fafbe35ce9004d5540e6" - integrity sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ== - parent-module@^1.0.0: version "1.0.1" resolved "https://registry.yarnpkg.com/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2" @@ -2306,23 +1884,11 @@ path-type@^4.0.0: resolved "https://registry.yarnpkg.com/path-type/-/path-type-4.0.0.tgz#84ed01c0a7ba380afe09d90a8c180dcd9d03043b" integrity sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw== -picocolors@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.0.0.tgz#cb5bdc74ff3f51892236eaf79d68bc44564ab81c" - integrity sha512-1fygroTLlHu66zi26VoTDv8yRgm0Fccecssto+MhsZ0D/DGW2sm8E8AjW7NU5VVTRt5GxbeZ5qBuJr+HyLYkjQ== - picomatch@^2.3.1: version "2.3.1" resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== -pkg-dir@^4.2.0: - version "4.2.0" - resolved "https://registry.yarnpkg.com/pkg-dir/-/pkg-dir-4.2.0.tgz#f099133df7ede422e81d1d8448270eeb3e4261f3" - integrity sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ== - dependencies: - find-up "^4.0.0" - plotly.js-dist@^2.30.1: version "2.30.1" resolved "https://registry.yarnpkg.com/plotly.js-dist/-/plotly.js-dist-2.30.1.tgz#98ef3f6e5d5f34d53390c2e8ff5d2668bfcfb4d8" @@ -2357,25 +1923,11 @@ queue-microtask@^1.2.2: resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== -randombytes@^2.1.0: - version "2.1.0" - resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a" - integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ== - dependencies: - safe-buffer "^5.1.0" - react-is@^16.13.1: version "16.13.1" resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4" integrity sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ== -rechoir@^0.8.0: - version "0.8.0" - resolved "https://registry.yarnpkg.com/rechoir/-/rechoir-0.8.0.tgz#49f866e0d32146142da3ad8f0eff352b3215ff22" - integrity sha512-/vxpCXddiX8NGfGO/mTafwjq4aFa/71pvamip0++IQk3zG8cbCj0fifNPrjjF1XMXUne91jL9OoxmdykoEtifQ== - dependencies: - resolve "^1.20.0" - reflect.getprototypeof@^1.0.4: version "1.0.6" resolved "https://registry.yarnpkg.com/reflect.getprototypeof/-/reflect.getprototypeof-1.0.6.tgz#3ab04c32a8390b770712b7a8633972702d278859" @@ -2404,24 +1956,12 @@ regexp.prototype.flags@^1.5.0, regexp.prototype.flags@^1.5.2: es-errors "^1.3.0" set-function-name "^2.0.1" -resolve-cwd@^3.0.0: - version "3.0.0" - resolved "https://registry.yarnpkg.com/resolve-cwd/-/resolve-cwd-3.0.0.tgz#0f0075f1bb2544766cf73ba6a6e2adfebcb13f2d" - integrity sha512-OrZaX2Mb+rJCpH/6CpSqt9xFVpN++x01XnN2ie9g6P5/3xelLAkXWVADpdz1IHD/KFfEXyE6V0U01OQ3UO2rEg== - dependencies: - resolve-from "^5.0.0" - resolve-from@^4.0.0: version "4.0.0" resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6" integrity sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g== -resolve-from@^5.0.0: - version "5.0.0" - resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-5.0.0.tgz#c35225843df8f776df21c57557bc087e9dfdfc69" - integrity sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw== - -resolve@^1.20.0, resolve@^1.22.0, resolve@^1.22.4: +resolve@^1.22.0, resolve@^1.22.4: version "1.22.8" resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.8.tgz#b6c87a9f2aa06dfab52e3d70ac8cde321fa5a48d" integrity sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw== @@ -2468,11 +2008,6 @@ safe-array-concat@^1.1.0, safe-array-concat@^1.1.2: has-symbols "^1.0.3" isarray "^2.0.5" -safe-buffer@^5.1.0: - version "5.2.1" - resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" - integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== - safe-regex-test@^1.0.3: version "1.0.3" resolved "https://registry.yarnpkg.com/safe-regex-test/-/safe-regex-test-1.0.3.tgz#a5b4c0f06e0ab50ea2c395c14d8371232924c377" @@ -2482,34 +2017,18 @@ safe-regex-test@^1.0.3: es-errors "^1.3.0" is-regex "^1.1.4" -schema-utils@^3.1.1, schema-utils@^3.2.0: - version "3.3.0" - resolved "https://registry.yarnpkg.com/schema-utils/-/schema-utils-3.3.0.tgz#f50a88877c3c01652a15b622ae9e9795df7a60fe" - integrity sha512-pN/yOAvcC+5rQ5nERGuwrjLlYvLTbCibnZ1I7B1LaiAz9BRBlE9GMgE/eqV30P7aJQUf7Ddimy/RsbYO/GrVGg== - dependencies: - "@types/json-schema" "^7.0.8" - ajv "^6.12.5" - ajv-keywords "^3.5.2" - semver@^6.3.0, semver@^6.3.1: version "6.3.1" resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== -semver@^7.3.4, semver@^7.3.7, semver@^7.5.4, semver@^7.6.0: +semver@^7.3.7, semver@^7.5.4, semver@^7.6.0: version "7.6.0" resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.0.tgz#1a46a4db4bffcccd97b743b5005c8325f23d4e2d" integrity sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg== dependencies: lru-cache "^6.0.0" -serialize-javascript@^6.0.1: - version "6.0.2" - resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-6.0.2.tgz#defa1e055c83bf6d59ea805d8da862254eb6a6c2" - integrity sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g== - dependencies: - randombytes "^2.1.0" - set-function-length@^1.2.1: version "1.2.2" resolved "https://registry.yarnpkg.com/set-function-length/-/set-function-length-1.2.2.tgz#aac72314198eaed975cf77b2c3b6b880695e5449" @@ -2532,13 +2051,6 @@ set-function-name@^2.0.0, set-function-name@^2.0.1: functions-have-names "^1.2.3" has-property-descriptors "^1.0.2" -shallow-clone@^3.0.0: - version "3.0.1" - resolved "https://registry.yarnpkg.com/shallow-clone/-/shallow-clone-3.0.1.tgz#8f2981ad92531f55035b01fb230769a40e02efa3" - integrity sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA== - dependencies: - kind-of "^6.0.2" - shebang-command@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" @@ -2576,24 +2088,6 @@ slash@^3.0.0: resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== -source-map-support@~0.5.20: - version "0.5.21" - resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.21.tgz#04fe7c7f9e1ed2d662233c28cb2b35b9f63f6e4f" - integrity sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w== - dependencies: - buffer-from "^1.0.0" - source-map "^0.6.0" - -source-map@^0.6.0: - version "0.6.1" - resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" - integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== - -source-map@^0.7.4: - version "0.7.4" - resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.7.4.tgz#a9bbe705c9d8846f4e08ff6765acf0f1b0898656" - integrity sha512-l3BikUxvPOcn5E74dZiq5BGsTb5yEwhaTSzccU6t4sDOH8NWJCstKO5QT2CvtFoK6F0saL7p9xHAqHOlCPJygA== - spdx-exceptions@^2.1.0: version "2.5.0" resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.5.0.tgz#5d607d27fc806f66d7b64a766650fa890f04ed66" @@ -2679,53 +2173,20 @@ supports-color@^7.1.0: dependencies: has-flag "^4.0.0" -supports-color@^8.0.0: - version "8.1.1" - resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c" - integrity sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q== - dependencies: - has-flag "^4.0.0" - supports-preserve-symlinks-flag@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09" integrity sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w== -tapable@^2.1.1, tapable@^2.2.0: - version "2.2.1" - resolved "https://registry.yarnpkg.com/tapable/-/tapable-2.2.1.tgz#1967a73ef4060a82f12ab96af86d52fdb76eeca0" - integrity sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ== - -terser-webpack-plugin@^5.3.10: - version "5.3.10" - resolved "https://registry.yarnpkg.com/terser-webpack-plugin/-/terser-webpack-plugin-5.3.10.tgz#904f4c9193c6fd2a03f693a2150c62a92f40d199" - integrity sha512-BKFPWlPDndPs+NGGCr1U59t0XScL5317Y0UReNrHaw9/FwhPENlq6bfgs+4yPfyP51vqC1bQ4rp1EfXW5ZSH9w== - dependencies: - "@jridgewell/trace-mapping" "^0.3.20" - jest-worker "^27.4.5" - schema-utils "^3.1.1" - serialize-javascript "^6.0.1" - terser "^5.26.0" - -terser@^5.26.0: - version "5.29.2" - resolved "https://registry.yarnpkg.com/terser/-/terser-5.29.2.tgz#c17d573ce1da1b30f21a877bffd5655dd86fdb35" - integrity sha512-ZiGkhUBIM+7LwkNjXYJq8svgkd+QK3UUr0wJqY4MieaezBSAIPgbSPZyIx0idM6XWK5CMzSWa8MJIzmRcB8Caw== - dependencies: - "@jridgewell/source-map" "^0.3.3" - acorn "^8.8.2" - commander "^2.20.0" - source-map-support "~0.5.20" - text-table@^0.2.0: version "0.2.0" resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw== -three@^0.158.0: - version "0.158.0" - resolved "https://registry.yarnpkg.com/three/-/three-0.158.0.tgz#03ddd5b60ce9c31be8fb69f27a6d9efd49908ac3" - integrity sha512-TALj4EOpdDPF1henk2Q+s17K61uEAAWQ7TJB68nr7FKxqwyDr3msOt5IWdbGm4TaWKjrtWS8DJJWe9JnvsWOhQ== +three@^0.163.0: + version "0.163.0" + resolved "https://registry.yarnpkg.com/three/-/three-0.163.0.tgz#cbfefbfd64a1353ab7cc8bf0fc396ddca1875a49" + integrity sha512-HlMgCb2TF/dTLRtknBnjUTsR8FsDqBY43itYop2+Zg822I+Kd0Ua2vs8CvfBVefXkBdNDrLMoRTGCIIpfCuDew== to-regex-range@^5.0.1: version "5.0.1" @@ -2739,17 +2200,6 @@ ts-api-utils@^1.0.1: resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-1.3.0.tgz#4b490e27129f1e8e686b45cc4ab63714dc60eea1" integrity sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ== -ts-loader@^9.5.1: - version "9.5.1" - resolved "https://registry.yarnpkg.com/ts-loader/-/ts-loader-9.5.1.tgz#63d5912a86312f1fbe32cef0859fb8b2193d9b89" - integrity sha512-rNH3sK9kGZcH9dYzC7CewQm4NtxJTjSEVRJ2DyBZR7f8/wcta+iV44UPCXc5+nzDzivKtlzV6c9P4e+oFhDLYg== - dependencies: - chalk "^4.1.0" - enhanced-resolve "^5.0.0" - micromatch "^4.0.0" - semver "^7.3.4" - source-map "^0.7.4" - tsconfig-paths@^3.14.1, tsconfig-paths@^3.15.0: version "3.15.0" resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz#5299ec605e55b1abb23ec939ef15edaf483070d4" @@ -2853,19 +2303,6 @@ unbox-primitive@^1.0.2: has-symbols "^1.0.3" which-boxed-primitive "^1.0.2" -undici-types@~5.26.4: - version "5.26.5" - resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617" - integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA== - -update-browserslist-db@^1.0.13: - version "1.0.13" - resolved "https://registry.yarnpkg.com/update-browserslist-db/-/update-browserslist-db-1.0.13.tgz#3c5e4f5c083661bd38ef64b6328c26ed6c8248c4" - integrity sha512-xebP81SNcPuNpPP3uzeW1NYXxI3rxyJzF3pD6sH4jE7o/IX+WtSpwnVU+qIsDPyk0d3hmFQ7mjqc6AtV604hbg== - dependencies: - escalade "^3.1.1" - picocolors "^1.0.0" - uri-js@^4.2.2: version "4.4.1" resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" @@ -2883,77 +2320,6 @@ vscode-textmate@^8.0.0: resolved "https://registry.yarnpkg.com/vscode-textmate/-/vscode-textmate-8.0.0.tgz#2c7a3b1163ef0441097e0b5d6389cd5504b59e5d" integrity sha512-AFbieoL7a5LMqcnOF04ji+rpXadgOXnZsxQr//r83kLPr7biP7am3g9zbaZIaBGwBRWeSvoMD4mgPdX3e4NWBg== -watchpack@^2.4.0: - version "2.4.1" - resolved "https://registry.yarnpkg.com/watchpack/-/watchpack-2.4.1.tgz#29308f2cac150fa8e4c92f90e0ec954a9fed7fff" - integrity sha512-8wrBCMtVhqcXP2Sup1ctSkga6uc2Bx0IIvKyT7yTFier5AXHooSI+QyQQAtTb7+E0IUCCKyTFmXqdqgum2XWGg== - dependencies: - glob-to-regexp "^0.4.1" - graceful-fs "^4.1.2" - -webpack-cli@^5.1.4: - version "5.1.4" - resolved "https://registry.yarnpkg.com/webpack-cli/-/webpack-cli-5.1.4.tgz#c8e046ba7eaae4911d7e71e2b25b776fcc35759b" - integrity sha512-pIDJHIEI9LR0yxHXQ+Qh95k2EvXpWzZ5l+d+jIo+RdSm9MiHfzazIxwwni/p7+x4eJZuvG1AJwgC4TNQ7NRgsg== - dependencies: - "@discoveryjs/json-ext" "^0.5.0" - "@webpack-cli/configtest" "^2.1.1" - "@webpack-cli/info" "^2.0.2" - "@webpack-cli/serve" "^2.0.5" - colorette "^2.0.14" - commander "^10.0.1" - cross-spawn "^7.0.3" - envinfo "^7.7.3" - fastest-levenshtein "^1.0.12" - import-local "^3.0.2" - interpret "^3.1.1" - rechoir "^0.8.0" - webpack-merge "^5.7.3" - -webpack-merge@^5.7.3: - version "5.10.0" - resolved "https://registry.yarnpkg.com/webpack-merge/-/webpack-merge-5.10.0.tgz#a3ad5d773241e9c682803abf628d4cd62b8a4177" - integrity sha512-+4zXKdx7UnO+1jaN4l2lHVD+mFvnlZQP/6ljaJVb4SZiwIKeUnrT5l0gkT8z+n4hKpC+jpOv6O9R+gLtag7pSA== - dependencies: - clone-deep "^4.0.1" - flat "^5.0.2" - wildcard "^2.0.0" - -webpack-sources@^3.2.3: - version "3.2.3" - resolved "https://registry.yarnpkg.com/webpack-sources/-/webpack-sources-3.2.3.tgz#2d4daab8451fd4b240cc27055ff6a0c2ccea0cde" - integrity sha512-/DyMEOrDgLKKIG0fmvtz+4dUX/3Ghozwgm6iPp8KRhvn+eQf9+Q7GWxVNMk3+uCPWfdXYC4ExGBckIXdFEfH1w== - -webpack@^5.90.3: - version "5.90.3" - resolved "https://registry.yarnpkg.com/webpack/-/webpack-5.90.3.tgz#37b8f74d3ded061ba789bb22b31e82eed75bd9ac" - integrity sha512-h6uDYlWCctQRuXBs1oYpVe6sFcWedl0dpcVaTf/YF67J9bKvwJajFulMVSYKHrksMB3I/pIagRzDxwxkebuzKA== - dependencies: - "@types/eslint-scope" "^3.7.3" - "@types/estree" "^1.0.5" - "@webassemblyjs/ast" "^1.11.5" - "@webassemblyjs/wasm-edit" "^1.11.5" - "@webassemblyjs/wasm-parser" "^1.11.5" - acorn "^8.7.1" - acorn-import-assertions "^1.9.0" - browserslist "^4.21.10" - chrome-trace-event "^1.0.2" - enhanced-resolve "^5.15.0" - es-module-lexer "^1.2.1" - eslint-scope "5.1.1" - events "^3.2.0" - glob-to-regexp "^0.4.1" - graceful-fs "^4.2.9" - json-parse-even-better-errors "^2.3.1" - loader-runner "^4.2.0" - mime-types "^2.1.27" - neo-async "^2.6.2" - schema-utils "^3.2.0" - tapable "^2.1.1" - terser-webpack-plugin "^5.3.10" - watchpack "^2.4.0" - webpack-sources "^3.2.3" - which-boxed-primitive@^1.0.2: version "1.0.2" resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.0.2.tgz#13757bc89b209b049fe5d86430e21cf40a89a8e6" @@ -3011,11 +2377,6 @@ which@^2.0.1: dependencies: isexe "^2.0.0" -wildcard@^2.0.0: - version "2.0.1" - resolved "https://registry.yarnpkg.com/wildcard/-/wildcard-2.0.1.tgz#5ab10d02487198954836b6349f74fff961e10f67" - integrity sha512-CC1bOL87PIWSBhDcTrdeLo6eGT7mCFtrg0uIJtqJUFyK+eJnzl8A1niH56uu7KMa5XFrtiV+AQuHO3n7DsHnLQ== - wrappy@1: version "1.0.2" resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f"