diff --git a/README.md b/README.md index 45591be7..953dfe85 100644 --- a/README.md +++ b/README.md @@ -1,5 +1,5 @@ ![The MRjs logo, an indigo and purple bowtie.](https://docs.mrjs.io/static/mrjs-logo.svg) - + An extensible library of Web Components for the spatial web. [![npm run build](https://github.com/Volumetrics-io/mrjs/actions/workflows/build.yml/badge.svg)](https://github.com/Volumetrics-io/mrjs/actions/workflows/build.yml) [![npm run test](https://github.com/Volumetrics-io/mrjs/actions/workflows/test.yml/badge.svg)](https://github.com/Volumetrics-io/mrjs/actions/workflows/test.yml) [![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg)](https://github.com/Volumetrics-io/mrjs/blob/main/LICENSE) diff --git a/__tests__/checkSubmoduleTesting.test.js b/__tests__/checkSubmoduleTesting.test.js index 6b4c3410..cdc507b2 100644 --- a/__tests__/checkSubmoduleTesting.test.js +++ b/__tests__/checkSubmoduleTesting.test.js @@ -1,36 +1,39 @@ -import { exec } from 'child_process'; +// import { exec } from 'child_process'; -// Function to promisify exec for easier use with async/await -const execPromise = (cmd) => - new Promise((resolve, reject) => { - exec(cmd, (error, stdout, stderr) => { - if (error) { - // Attach stdout and stderr to the error object for better debugging - error.stdout = stdout; - error.stderr = stderr; - reject(error); - } else { - resolve({ stdout, stderr, code: 0 }); // Explicitly resolve with code 0 for success - } - }); - }); +// // Function to promisify exec for easier use with async/await +// const execPromise = (cmd) => +// new Promise((resolve, reject) => { +// exec(cmd, (error, stdout, stderr) => { +// if (error) { +// // Attach stdout and stderr to the error object for better debugging +// error.stdout = stdout; +// error.stderr = stderr; +// reject(error); +// } else { +// resolve({ stdout, stderr, code: 0 }); // Explicitly resolve with code 0 for success +// } +// }); +// }); // Check that the mrjsio submodule is up to date // - note: doing this as a test instead of another github action is that // it's a simpler process to ask the user to update following the // test failure considering it's more locally dependent. test('mrjsio submodule is up to date', async () => { - try { - const result = await execPromise('./scripts/check-if-submodule-needs-update.sh ./samples/mrjsio'); - // If the promise resolves, it means the script exited with code 0 - expect(result.code).toBe(0); - } catch (err) { - // If the script exits with a non-zero exit code, it will be caught here - console.error('Script failed to execute:', err.stderr); - console.log('!!! mrjsio submodule needs to be updated !!! run: `npm run update-submodules` and it will handle the rest for you :)'); + // Need to comment out this testing file as it causes an improper merge/stash/branch setup when run with + // the rest of the tests automatically. Will bring it back as part of #608 being resolved + console.log('SKIPPING THIS FOR NOW - see pending issue: https://github.com/Volumetrics-io/mrjs/issues/608'); +// try { +// const result = await execPromise('./scripts/check-if-submodule-needs-update.sh ./samples/mrjsio'); +// // If the promise resolves, it means the script exited with code 0 +// expect(result.code).toBe(0); +// } catch (err) { +// // If the script exits with a non-zero exit code, it will be caught here +// console.error('Script failed to execute:', err.stderr); +// console.log('!!! mrjsio submodule needs to be updated !!! run: `npm run update-submodules` and it will handle the rest for you :)'); - // Fail the test by checking the exit code - since success is 0, checking against - // 0 is guaranteed to trigger a failure. - expect(err.code).toBe(0); - } +// // Fail the test by checking the exit code - since success is 0, checking against +// // 0 is guaranteed to trigger a failure. +// expect(err.code).toBe(0); +// } }); diff --git a/__tests__/examplesTesting.test.js b/__tests__/examplesTesting.test.js index b744cbc8..fc3a22e0 100644 --- a/__tests__/examplesTesting.test.js +++ b/__tests__/examplesTesting.test.js @@ -1,5 +1,5 @@ import * as puppeteer from 'puppeteer'; -import fs from 'fs/promises'; // Node.js file system module with promises +import fs from 'fs/promises'; // todo: in future dont hard code this, but the relative links based on filepath dont work // using a server to host them works best, so just grabbing from github is fine for now. @@ -14,19 +14,19 @@ describe('Test the Examples', () => { browser = await puppeteer.launch({ headless: true }); page = await browser.newPage(); - // Listen for console errors right after creating the page page.on('console', msg => { if (msg.type() === 'error') { errors.push(msg.text()); console.error(`Console error: ${msg.text()}`); + } else { + console.log('PAGE LOG:', msg.text()); } }); - // Catch unhandled promise rejections - page.on('pageerror', error => { - errors.push(error.toString()); - console.error(`Unhandled error: ${error}`); - }); + // page.on('pageerror', error => { + // errors.push(error.toString()); + // console.error(`Unhandled error: ${error}`); + // }); }); afterAll(async () => { @@ -35,24 +35,21 @@ describe('Test the Examples', () => { fileNames.forEach(fileName => { test(`Page ${fileName} should load with no console errors`, async () => { - // Reset errors array for each file errors = []; let htmlContent = await fs.readFile(`./dist/examples/${fileName}.html`, 'utf8'); console.log(`Running test on: ./dist/examples/${fileName}.html`); - // Adjust script and link paths - htmlContent = htmlContent.replace( - ``, - ``); - htmlContent = htmlContent.replace( - ``, - ``); - + // Adjust script path to load mr.js relatively, index.html is in propert spot already + if (fileName != "../index") { + htmlContent = htmlContent.replace( + ``, + `` + ); + } + await page.setContent(htmlContent); - await page.waitForTimeout(1000); // wait for a second to allow all scripts to execute - // Assertions can be placed here if needed expect(errors).toHaveLength(0); }); }); diff --git a/__tests__/unitTestingBase.test.txt b/__tests__/unitTestingBase.test.txt deleted file mode 100644 index 7ddaa007..00000000 --- a/__tests__/unitTestingBase.test.txt +++ /dev/null @@ -1,58 +0,0 @@ -// import fetchMock from 'jest-fetch-mock'; - -// import { Model } from '../src/utils/Model'; - -// fetchMock.enableMocks(); - -// const MRJS_TESTING_SERVER = 'https://localhost:8080'; - -// // todo: in future dont hard code this, but the relative links based on filepath dont work -// // using a server to host them works best, so just grabbing from github is fine for now. -// const MODELS_URL = 'https://github.com/Volumetrics-io/MR.js/blob/main/assets/models/'; - -// describe('loadModel function tests', () => { -// beforeAll(async () => { -// page = await browser.newPage(); -// await page.goto(MRJS_TESTING_SERVER); -// }); - -// test('checkLoadModel - stl', async () => { -// // Mock the network response for STL model -// fetchMock.mockResponseOnce('mocked STL model response'); - -// // Call your loadModel function with Puppeteer -// const response = await page.evaluate(() => -// Model.loadModel('${MODELS_URL}logo.stl', 'stl') -// ); - -// expect(response).toBe('mocked STL model response'); // Update assertion as per your logic -// }); - -// test('checkLoadModel - glb', async () => { -// // Mock the network response for GLB model -// fetchMock.mockResponseOnce('mocked GLB model response'); - -// // Call your loadModel function with Puppeteer -// const response = await page.evaluate(() => -// Model.loadModel('${MODELS_URL}logo.glb', 'glb') -// ); - -// expect(response).toBe('mocked GLB model response'); // Update assertion as per your logic -// }); - -// test('checkLoadModel - fbx', async () => { -// // Mock the network response for FBX model -// fetchMock.mockResponseOnce('mocked FBX model response'); - -// // Call your loadModel function with Puppeteer -// const response = await page.evaluate(() => -// Model.loadModel('${MODELS_URL}logo.fbx', 'fbx') -// ); - -// expect(response).toBe('mocked FBX model response'); // Update assertion as per your logic -// }); - -// // Additional tests for GLB and FBX... -// }); - -// commenting out for now diff --git a/dist/mr.js b/dist/mr.js index 466149b0..332ed898 100644 --- a/dist/mr.js +++ b/dist/mr.js @@ -59,7 +59,7 @@ eval("\n\nmodule.exports = function (i) {\n return i[1];\n};\n\n//# sourceURL=w \**********************/ /***/ ((module) => { -eval("module.exports = {\"name\":\"mrjs\",\"version\":\"0.6.2\",\"type\":\"module\",\"description\":\"an MR first webXR framework\",\"engines\":{\"node\":\"^18.17.0 || >=20.5.0\"},\"main\":\"dist/mr.js\",\"homepage\":\"https://mrjs.io\",\"scripts\":{\"build\":\"npx webpack --config webpack.config.js\",\"update-submodules\":\"./scripts/update-all-submodules.sh\",\"server\":\"npx webpack serve\",\"test-server\":\"NODE_ENV=testing npx webpack serve\",\"test\":\"NODE_OPTIONS=--experimental-vm-modules npx jest\",\"test-serially\":\"NODE_OPTIONS=--experimental-vm-modules npx jest --runInBand\",\"test-randomized\":\"NODE_OPTIONS=--experimental-vm-modules npx jest --runInBand --random\",\"docs\":\"./scripts/create-docs.sh\",\"prettier-check\":\"prettier --check \\\"src/**/*.js\\\" \\\"*.js\\\"\",\"prettier-fix\":\"prettier --write \\\"src/**/*.js\\\" \\\"*.js\\\"\",\"lint-check\":\"eslint \\\"src/**/*.js\\\" \\\"*.js\\\" --ignore-pattern \\\"src/extras/**\\\" --max-warnings 0\",\"lint-fix\":\"eslint \\\"src/**/*.js\\\" \\\"*.js\\\" --ignore-pattern \\\"src/extras/**\\\" --fix\",\"format\":\"npm run prettier-fix && npm run lint-fix\",\"check-format\":\"npm run prettier-check && npm run lint-check\"},\"repository\":{\"type\":\"git\",\"url\":\"git+https://github.com/Volumetrics-io/mrjs.git\"},\"author\":\"Volumetrics\",\"license\":\"MIT\",\"bugs\":{\"url\":\"https://github.com/Volumetrics-io/mrjs/issues\"},\"testMatch\":[\"**/__tests__/**/*.test.mjs\"],\"testEnvironment\":\"node\",\"devDependencies\":{\"@babel/core\":\"^7.23.9\",\"babel-jest\":\"^29.7.0\",\"copy-webpack-plugin\":\"^11.0.0\",\"css-loader\":\"^6.10.0\",\"eslint\":\"^8.56.0\",\"eslint-config-airbnb\":\"^19.0.4\",\"eslint-config-airbnb-base\":\"^15.0.0\",\"eslint-config-prettier\":\"^8.10.0\",\"eslint-plugin-import\":\"^2.29.1\",\"eslint-plugin-jsdoc\":\"^46.10.1\",\"eslint-plugin-jsx-a11y\":\"^6.8.0\",\"eslint-plugin-prettier\":\"^4.2.1\",\"eslint-plugin-react\":\"^7.33.2\",\"eslint-plugin-react-hooks\":\"^4.6.0\",\"eslint-plugin-unused-imports\":\"^3.0.0\",\"esm\":\"^3.2.25\",\"html-webpack-plugin\":\"^5.6.0\",\"install\":\"^0.13.0\",\"jest\":\"^29.7.0\",\"jest-environment-jsdom\":\"^29.7.0\",\"jest-fetch-mock\":\"^3.0.3\",\"jest-puppeteer\":\"^9.0.2\",\"jsdoc\":\"^4.0.2\",\"jsdoc-to-markdown\":\"^8.0.1\",\"json-loader\":\"^0.5.7\",\"mini-css-extract-plugin\":\"^2.8.0\",\"npm\":\"^10.4.0\",\"playwright\":\"^1.41.2\",\"puppeteer\":\"^21.10.0\",\"style-loader\":\"^3.3.4\",\"tui-jsdoc-template\":\"^1.2.2\",\"url\":\"^0.11.3\",\"webpack\":\"^5.90.1\",\"webpack-cli\":\"^5.1.4\",\"webpack-dev-server\":\"^4.15.1\"},\"dependencies\":{\"@babel/eslint-parser\":\"^7.23.10\",\"@dimforge/rapier3d\":\"^0.12.0\",\"docdash\":\"^2.0.2\",\"jaguarjs-jsdoc\":\"^1.1.0\",\"jsdom\":\"^23.2.0\",\"prettier\":\"^3.2.4\",\"prettier-eslint-cli\":\"^8.0.1\",\"stats.js\":\"^0.17.0\",\"three\":\"^0.161.0\",\"troika-three-text\":\"^0.48.1\"}}\n\n//# sourceURL=webpack://mrjs/./package.json?"); +eval("module.exports = {\"name\":\"mrjs\",\"version\":\"0.6.2\",\"type\":\"module\",\"description\":\"an MR first webXR framework\",\"engines\":{\"node\":\"^18.17.0 || >=20.5.0\"},\"main\":\"dist/mr.js\",\"homepage\":\"https://mrjs.io\",\"scripts\":{\"build\":\"npx webpack --config webpack.config.js\",\"update-submodules\":\"./scripts/update-all-submodules.sh\",\"server\":\"npx webpack serve\",\"test-server\":\"NODE_ENV=testing npx webpack serve\",\"test\":\"NODE_OPTIONS=--experimental-vm-modules npx jest\",\"test-serially\":\"NODE_OPTIONS=--experimental-vm-modules npx jest --runInBand\",\"test-randomized\":\"NODE_OPTIONS=--experimental-vm-modules npx jest --runInBand --random\",\"clear-testing-cache\":\"npx jest --clearCache\",\"docs\":\"./scripts/create-docs.sh\",\"prettier-check\":\"prettier --check \\\"src/**/*.js\\\" \\\"*.js\\\"\",\"prettier-fix\":\"prettier --write \\\"src/**/*.js\\\" \\\"*.js\\\"\",\"lint-check\":\"eslint \\\"src/**/*.js\\\" \\\"*.js\\\" --ignore-pattern \\\"src/extras/**\\\" --max-warnings 0\",\"lint-fix\":\"eslint \\\"src/**/*.js\\\" \\\"*.js\\\" --ignore-pattern \\\"src/extras/**\\\" --fix\",\"format\":\"npm run prettier-fix && npm run lint-fix\",\"check-format\":\"npm run prettier-check && npm run lint-check\"},\"repository\":{\"type\":\"git\",\"url\":\"git+https://github.com/Volumetrics-io/mrjs.git\"},\"author\":\"Volumetrics\",\"license\":\"MIT\",\"bugs\":{\"url\":\"https://github.com/Volumetrics-io/mrjs/issues\"},\"testMatch\":[\"**/__tests__/**/*.test.mjs\"],\"testEnvironment\":\"node\",\"devDependencies\":{\"@babel/core\":\"^7.23.9\",\"babel-jest\":\"^29.7.0\",\"copy-webpack-plugin\":\"^11.0.0\",\"css-loader\":\"^6.10.0\",\"eslint\":\"^8.56.0\",\"eslint-config-airbnb\":\"^19.0.4\",\"eslint-config-airbnb-base\":\"^15.0.0\",\"eslint-config-prettier\":\"^8.10.0\",\"eslint-plugin-import\":\"^2.29.1\",\"eslint-plugin-jsdoc\":\"^46.10.1\",\"eslint-plugin-jsx-a11y\":\"^6.8.0\",\"eslint-plugin-prettier\":\"^4.2.1\",\"eslint-plugin-react\":\"^7.33.2\",\"eslint-plugin-react-hooks\":\"^4.6.0\",\"eslint-plugin-unused-imports\":\"^3.0.0\",\"esm\":\"^3.2.25\",\"html-webpack-plugin\":\"^5.6.0\",\"install\":\"^0.13.0\",\"jest\":\"^29.7.0\",\"jest-environment-jsdom\":\"^29.7.0\",\"jest-fetch-mock\":\"^3.0.3\",\"jest-puppeteer\":\"^9.0.2\",\"jsdoc\":\"^4.0.2\",\"jsdoc-to-markdown\":\"^8.0.1\",\"json-loader\":\"^0.5.7\",\"mini-css-extract-plugin\":\"^2.8.0\",\"npm\":\"^10.4.0\",\"playwright\":\"^1.41.2\",\"puppeteer\":\"^21.10.0\",\"style-loader\":\"^3.3.4\",\"tui-jsdoc-template\":\"^1.2.2\",\"url\":\"^0.11.3\",\"webpack\":\"^5.90.1\",\"webpack-cli\":\"^5.1.4\",\"webpack-dev-server\":\"^4.15.1\"},\"dependencies\":{\"@babel/eslint-parser\":\"^7.23.10\",\"@dimforge/rapier3d\":\"^0.12.0\",\"docdash\":\"^2.0.2\",\"jaguarjs-jsdoc\":\"^1.1.0\",\"jsdom\":\"^23.2.0\",\"prettier\":\"^3.2.4\",\"prettier-eslint-cli\":\"^8.0.1\",\"stats.js\":\"^0.17.0\",\"three\":\"^0.161.0\",\"troika-three-text\":\"^0.48.1\"}}\n\n//# sourceURL=webpack://mrjs/./package.json?"); /***/ }), @@ -487,7 +487,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ AnchorSystem: () => (/* binding */ AnchorSystem)\n/* harmony export */ });\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n/* harmony import */ var mrjs_core_MRSystem__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs/core/MRSystem */ \"./src/core/MRSystem.js\");\n/* harmony import */ var mrjs_dataManagers_MRPlaneManager__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! mrjs/dataManagers/MRPlaneManager */ \"./src/dataManagers/MRPlaneManager.js\");\n\n\n\n\n\n\n\n/**\n * @class AnchorSystem\n * @classdesc creates and manages WebXR anchors in the MR scene.\n * @augments MRSystem\n */\nclass AnchorSystem extends mrjs_core_MRSystem__WEBPACK_IMPORTED_MODULE_1__.MRSystem {\n /**\n * @class\n * @description AnchorSystem's default constructor including setting up event listeners for XR initialization, user interaction, and the MRPlaneManager\n */\n constructor() {\n super();\n this.sourceRequest = false;\n this.source;\n this.currentEntity = null;\n this.tempMatrix = new three__WEBPACK_IMPORTED_MODULE_3__.Matrix4();\n\n this.planeManager = new mrjs_dataManagers_MRPlaneManager__WEBPACK_IMPORTED_MODULE_2__.MRPlaneManager(this.app.scene, this.app.getAttribute('occlusion'));\n this.anchoringQueue = new Set();\n\n this.hitResults;\n\n this.userWorldPosition = new three__WEBPACK_IMPORTED_MODULE_3__.Vector3();\n this.cameraForward = new three__WEBPACK_IMPORTED_MODULE_3__.Vector3();\n this.pinchDistance = 0;\n\n this.axisSwapQuat = new three__WEBPACK_IMPORTED_MODULE_3__.Quaternion().setFromAxisAngle(new three__WEBPACK_IMPORTED_MODULE_3__.Vector3(1, 0, 0), -(3 * Math.PI) / 2);\n\n this.hand = null;\n\n this.snapDistance = 0.6;\n\n this.scale = 1;\n\n let existing = document.querySelectorAll('[data-comp-anchor]');\n\n this.originPosition = new three__WEBPACK_IMPORTED_MODULE_3__.Vector3();\n\n for (const entity of existing) {\n this.attachedComponent(entity);\n this.registry.add(entity);\n }\n\n this.app.addEventListener('enterxr', () => {\n if (this.sourceRequest == false) {\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestReferenceSpace('viewer').then((viewerSpace) => {\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestHitTestSource({ space: viewerSpace }).then((source) => {\n this.source = source;\n });\n });\n\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.addEventListener('end', () => {\n this.sourceRequest = false;\n this.source = null;\n this.hand = null;\n });\n\n this.sourceRequest = true;\n }\n });\n\n this.app.addEventListener('exitxr', () => {\n this.deleteAnchor(this.app);\n this.app.origin.matrix.copy(new three__WEBPACK_IMPORTED_MODULE_3__.Matrix4());\n });\n\n document.addEventListener('selectstart', (event) => {\n if (this.currentEntity == null || (this.hand && this.hand != event.detail.handedness)) {\n return;\n }\n if (!event.detail) {\n return;\n }\n this.hand = event.detail.handedness;\n });\n\n document.addEventListener('selectmoved', (event) => {\n if (this.currentEntity && this.hand == event.detail.handedness) {\n this.userWorldPosition.setFromMatrixPosition(this.app.camera.matrixWorld);\n this.cameraForward.setFromMatrixPosition(this.app.user.forward.matrixWorld);\n\n this.pinchDistance = this.cameraForward.distanceTo(event.detail.position);\n this.scale = Math.exp(2 * this.pinchDistance);\n this.app.anchor.position.z = this.app.user.forward.position.z * this.scale;\n this.app.anchor.lookAt(this.userWorldPosition);\n }\n });\n\n document.addEventListener('selectend', (event) => {\n if (this.currentEntity == null || this.hand == null || this.hand != event.detail.handedness) {\n return;\n }\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n frame.createAnchor(this.matrix4ToXRRigidTransform(this.currentEntity.object3D.matrixWorld), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n this.currentEntity.anchor = anchor;\n this.anchoringQueue.delete(this.currentEntity);\n this.currentEntity = null;\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n this.hand = null;\n });\n }\n\n /**\n * @function\n * @description This update function maintains the transforms of anchored entities.\n * This overrides any other transform values set on the element when in mixed reality.\n * @param {number} deltaTime - given timestep to be used for any feature changes\n * @param {object} frame - given frame information to be used for any feature changes\n */\n update(deltaTime, frame) {\n if (mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.isPresenting) {\n if (this.currentEntity) {\n this.floating(frame);\n }\n\n if (!this.app.anchor) {\n this.setAppOrigin();\n } else {\n this.updateOrigin(frame);\n }\n }\n\n for (const entity of this.registry) {\n if (mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.isPresenting) {\n let anchorComp = entity.components.get('anchor');\n if (entity.anchor == null && !this.anchoringQueue.has(entity)) {\n entity.object3D.matrixWorldAutoUpdate = false;\n this.createAnchor(entity, anchorComp);\n } else if (entity.anchor) {\n let pose = frame.getPose(entity.anchor.anchorSpace, mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace);\n let transform = this.multiplyQuaternionWithXRRigidTransform(this.axisSwapQuat, pose.transform);\n\n entity.object3D.matrix.copy(this.adjustTransform(transform));\n } else {\n this.createAnchor(entity, anchorComp);\n }\n } else if (entity.anchor) {\n entity.object3D.matrix.copy(entity.object3D.userData.originalMatrix);\n this.deleteAnchor(entity);\n }\n }\n }\n\n /**\n * @function\n * @description Called when the entity component is initialized\n * @param {object} entity - the entity being attached/initialized.\n */\n attachedComponent(entity) {\n entity.object3D.userData.originalMatrix = new three__WEBPACK_IMPORTED_MODULE_3__.Matrix4();\n entity.object3D.userData.originalMatrix.copy(entity.object3D.matrixWorld);\n entity.object3D.matrixAutoUpdate = false;\n }\n\n /**\n * @function\n * @description Called when the entity component is updated\n * @param {object} entity - the entity being updated based on the component.\n */\n updatedComponent(entity) {\n // delete before creating a new one.\n this.deleteAnchor(entity);\n // // These cases can be managed instantly\n // this.createAnchor(entity, comp);\n }\n\n /**\n * @function\n * @description Called when the entity component is detached\n * @param {object} entity - the entity being updated based on the component being detached.\n */\n detachedComponent(entity) {\n entity.object3D.matrixAutoUpdate = true;\n this.deleteAnchor(entity);\n }\n\n /**\n * @function\n * @description deletes anchors from the scene and removes all references to the anchored plane (if any)\n * @param {object} entity - the entity whose anchor is being deleted.\n */\n deleteAnchor(entity) {\n if (entity.plane) {\n entity.plane.occupied = false;\n entity.plane.mesh.visible = true;\n entity.plane = null;\n }\n entity.anchor = null;\n entity.dispatchEvent(new CustomEvent('anchorremoved', { bubbles: true }));\n }\n\n /**\n * @function\n * @description creates the anchor specified by the data-anchor-comp\n * @param {object} entity - the entity whose anchor is being created.\n * @param {object} comp - the data component with a type value that represents the string 'fixed', 'plane', 'floating', etc\n */\n createAnchor(entity, comp) {\n switch (comp.type) {\n case 'fixed':\n this.fixed(entity);\n break;\n case 'plane':\n this.plane(entity, comp);\n break;\n case 'floating':\n this.currentEntity = entity;\n this.anchoringQueue.add(entity);\n break;\n default:\n break;\n }\n }\n\n /**\n * @function\n * @description Sets the origin of the MRApp being touched by all systems to allow anchoring to position\n * itself properly.\n */\n setAppOrigin() {\n if (!mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.isPresenting) {\n return;\n }\n let originMatrix = new three__WEBPACK_IMPORTED_MODULE_3__.Matrix4();\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n originMatrix.copyPosition(this.app.user.forward.matrixWorld);\n frame.createAnchor(this.matrix4ToXRRigidTransform(originMatrix), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n this.app.origin.matrixAutoUpdate = false;\n this.app.anchor = anchor;\n this.app.dispatchEvent(new CustomEvent('anchored', { bubbles: true }));\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n }\n\n /**\n * @function\n * @description Updates the origin of the MRApp being touched by all systems to allow anchoring to position.\n * @param {object} frame - given frame information to be used for any feature changes (from the update(..) loop)\n */\n updateOrigin(frame) {\n let pose = frame.getPose(this.app.anchor.anchorSpace, mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace);\n let transform = this.multiplyQuaternionWithXRRigidTransform(this.axisSwapQuat, pose.transform);\n\n this.app.origin.matrix.copy(this.adjustTransform(transform, true));\n\n this.originPosition.setFromMatrixPosition(this.app.origin.matrixWorld);\n }\n\n /**\n * @function\n * @description Anchors the given entity half a meter in front of the users position at launch.\n * @param {object} entity - the entity being positioned.\n */\n fixed(entity) {\n this.anchoringQueue.add(entity);\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n frame.createAnchor(this.matrix4ToXRRigidTransform(this.app.user.forward.matrixWorld), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n entity.anchor = anchor;\n entity.dispatchEvent(new CustomEvent('anchored', { bubbles: true }));\n this.anchoringQueue.delete(entity);\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n }\n\n /**\n * @function\n * @description Creates an anchor at the position specified by the user,\n * either floating in front of them or pinned to the scene mesh\n * @param {object} frame - given frame information to be used for any feature changes (from the update(..) loop)\n */\n floating(frame) {\n this.hitResults = frame.getHitTestResults(this.source);\n const hit = this.hitResults[0];\n const pose = hit?.getPose(mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace);\n\n if (pose && this.userWorldPosition.distanceTo(pose.transform.position) < this.snapDistance * this.scale) {\n let transform = this.multiplyQuaternionWithXRRigidTransform(this.axisSwapQuat, pose.transform);\n this.currentEntity.object3D.matrix.copy(this.adjustTransform(transform));\n } else {\n this.currentEntity.object3D.matrix.copy(this.app.anchor.matrixWorld);\n }\n }\n\n /**\n * @function\n * @description Anchors the provided entity to the nearest unoccupied plane that meets the given orientation and label.\n * each plane is currently limited to one anchor for simplicity.\n * @param {object} entity - the entity being anchored by this function.\n * @param {object} comp - the data-component to determine the orientation and label of the associated plane\n */\n plane(entity, comp) {\n this.anchoringQueue.add(entity);\n this.userWorldPosition.setFromMatrixPosition(this.app.user.forward.matrixWorld);\n let sort = Array.from(this.planeManager.planeDictionary[comp.label].values());\n sort.sort((a, b) => {\n return a.mesh.position.distanceTo(this.userWorldPosition) - b.mesh.position.distanceTo(this.userWorldPosition);\n });\n for (const mrPlane of sort) {\n if (mrPlane.occupied) {\n continue;\n }\n if (comp.label && comp.label != mrPlane.label) {\n continue;\n }\n if (comp.orientation && comp.orientation != mrPlane.orientation) {\n continue;\n }\n mrPlane.occupied = true;\n\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n frame.createAnchor(this.matrix4ToXRRigidTransform(mrPlane.mesh.matrixWorld), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n this.anchoringQueue.delete(entity);\n if (!this.planeManager.planeDictionary[comp.label].has(mrPlane)) {\n return;\n }\n if (entity.anchor) {\n return;\n }\n entity.anchor = anchor;\n entity.plane = mrPlane;\n entity.dispatchEvent(new CustomEvent('anchored', { bubbles: true }));\n if (comp.occlusion == false) {\n mrPlane.mesh.visible = false;\n }\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n return;\n }\n }\n\n originalAnchorMatrix = new three__WEBPACK_IMPORTED_MODULE_3__.Matrix4();\n anchorForwardVector = new three__WEBPACK_IMPORTED_MODULE_3__.Vector3(0, 0, 0.03);\n rotationMatrix = new three__WEBPACK_IMPORTED_MODULE_3__.Matrix4();\n adjustedMatrix = new three__WEBPACK_IMPORTED_MODULE_3__.Matrix4();\n\n /**\n * @function\n * @description converts the provided XRRigidTransform to a Matrix4 and adjusts it to ensure\n * that it's y-axis is pointing directly up and it's z-axis is facing inward\n * @param {object} xrRigidTransform - a THREE.js transformation matrix that we want to adjust\n * @param {boolean} origin - true if this is positioned at the origin or not (handles special case of div-0).\n * @returns {object} a new adjusted THREE.js Matrix4\n */\n adjustTransform(xrRigidTransform, origin = false) {\n // Create a Three.js Quaternion for the XRRigidTransform's orientation\n let quaternion = new three__WEBPACK_IMPORTED_MODULE_3__.Quaternion(xrRigidTransform.orientation.x, xrRigidTransform.orientation.y, xrRigidTransform.orientation.z, xrRigidTransform.orientation.w);\n\n // Create a Three.js Vector for the up direction\n let upVector = new three__WEBPACK_IMPORTED_MODULE_3__.Vector3(0, 1, 0);\n\n // Apply the quaternion to the up direction\n let transformedUp = upVector.clone().applyQuaternion(quaternion);\n\n // Calculate the rotation needed to align the transformed up direction with the global up direction\n let correctionQuaternion = new three__WEBPACK_IMPORTED_MODULE_3__.Quaternion().setFromUnitVectors(transformedUp, upVector);\n\n // Apply the correction to the original quaternion\n quaternion.premultiply(correctionQuaternion);\n\n // Create a new Three.js Vector3 for the position\n let position = new three__WEBPACK_IMPORTED_MODULE_3__.Vector3(xrRigidTransform.position.x, xrRigidTransform.position.y, xrRigidTransform.position.z);\n\n if (!origin) {\n position.sub(this.originPosition);\n }\n\n this.originalAnchorMatrix.compose(position, quaternion, new three__WEBPACK_IMPORTED_MODULE_3__.Vector3(1, 1, 1));\n this.rotationMatrix.extractRotation(this.originalAnchorMatrix);\n\n this.anchorForwardVector.applyMatrix4(this.rotationMatrix);\n this.adjustedMatrix.makeTranslation(this.anchorForwardVector.x, this.anchorForwardVector.y, this.anchorForwardVector.y);\n this.anchorForwardVector.set(0, 0, 0.03);\n this.adjustedMatrix.multiply(this.originalAnchorMatrix);\n\n return this.adjustedMatrix;\n }\n\n /**\n * @function\n * @description Converts the provided matrix4 into a webXR xompatible XRRigidTransform\n * @param {object} matrix4 - the matrix we want to convert to a XRRigidTransform\n * @returns {object} xrRigidTransform - the converted representation of the param matrix4\n */\n matrix4ToXRRigidTransform(matrix4) {\n // Extract the translation component from the Matrix4\n const position = new three__WEBPACK_IMPORTED_MODULE_3__.Vector3();\n position.setFromMatrixPosition(matrix4);\n\n // Extract the rotation component from the Matrix4\n const quaternion = new three__WEBPACK_IMPORTED_MODULE_3__.Quaternion();\n quaternion.setFromRotationMatrix(matrix4);\n\n // Create a DOMPointInit for the position\n const positionInit = { x: position.x, y: position.y, z: position.z, w: 1 };\n\n // Create a DOMPointInit for the orientation (quaternion)\n const orientationInit = { x: quaternion.x, y: quaternion.y, z: quaternion.z, w: quaternion.w };\n\n // Create and return the XRRigidTransform\n return new XRRigidTransform(positionInit, orientationInit);\n }\n\n /**\n * @function\n * @description Multiplies an xr rigid transform by the provided quaternion\n * @param {object} quaternion - the quaternion we want to multiply with the xrRigidTransform\n * @param {object} xrRigidTransform - the second part of the multiplication we are looking to perform.\n * @returns {object} xrRigidTransform - the output of the quaternion * xrRigidTransform in the form of an xrRigidTransform\n */\n multiplyQuaternionWithXRRigidTransform(quaternion, xrRigidTransform) {\n // Create a Three.js Quaternion for the XRRigidTransform's orientation\n let transformQuaternion = new three__WEBPACK_IMPORTED_MODULE_3__.Quaternion(\n xrRigidTransform.orientation.x,\n xrRigidTransform.orientation.y,\n xrRigidTransform.orientation.z,\n xrRigidTransform.orientation.w\n );\n\n // Multiply the quaternions\n transformQuaternion.multiply(quaternion);\n\n // Create a new XRRigidTransform with the multiplied orientation and original position\n let newPosition = xrRigidTransform.position;\n let newOrientation = new DOMPointReadOnly(transformQuaternion.x, transformQuaternion.y, transformQuaternion.z, transformQuaternion.w);\n\n return new XRRigidTransform(newPosition, newOrientation);\n }\n}\n\n\n//# sourceURL=webpack://mrjs/./src/core/componentSystems/AnchorSystem.js?"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ AnchorSystem: () => (/* binding */ AnchorSystem)\n/* harmony export */ });\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n/* harmony import */ var mrjs_core_MREntity__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs/core/MREntity */ \"./src/core/MREntity.js\");\n/* harmony import */ var mrjs_core_MRSystem__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! mrjs/core/MRSystem */ \"./src/core/MRSystem.js\");\n/* harmony import */ var mrjs_dataManagers_MRPlaneManager__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! mrjs/dataManagers/MRPlaneManager */ \"./src/dataManagers/MRPlaneManager.js\");\n\n\n\n\n\n\n\n\n/**\n * @class AnchorSystem\n * @classdesc creates and manages WebXR anchors in the MR scene.\n * @augments MRSystem\n */\nclass AnchorSystem extends mrjs_core_MRSystem__WEBPACK_IMPORTED_MODULE_2__.MRSystem {\n /**\n * @class\n * @description AnchorSystem's default constructor including setting up event listeners for XR initialization, user interaction, and the MRPlaneManager\n */\n constructor() {\n super();\n this.sourceRequest = false;\n this.source;\n this.currentEntity = null;\n this.tempMatrix = new three__WEBPACK_IMPORTED_MODULE_4__.Matrix4();\n\n this.planeManager = new mrjs_dataManagers_MRPlaneManager__WEBPACK_IMPORTED_MODULE_3__.MRPlaneManager(this.app.scene, this.app.getAttribute('occlusion'));\n this.anchoringQueue = new Set();\n\n this.hitResults;\n\n this.userWorldPosition = new three__WEBPACK_IMPORTED_MODULE_4__.Vector3();\n this.cameraForward = new three__WEBPACK_IMPORTED_MODULE_4__.Vector3();\n this.pinchDistance = 0;\n\n this.axisSwapQuat = new three__WEBPACK_IMPORTED_MODULE_4__.Quaternion().setFromAxisAngle(new three__WEBPACK_IMPORTED_MODULE_4__.Vector3(1, 0, 0), -(3 * Math.PI) / 2);\n\n this.hand = null;\n\n this.snapDistance = 0.6;\n\n this.scale = 1;\n\n let existing = document.querySelectorAll('[data-comp-anchor]');\n\n this.originPosition = new three__WEBPACK_IMPORTED_MODULE_4__.Vector3();\n\n for (const entity of existing) {\n this.attachedComponent(entity);\n this.registry.add(entity);\n }\n\n this.app.addEventListener('enterxr', () => {\n if (this.sourceRequest == false) {\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestReferenceSpace('viewer').then((viewerSpace) => {\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestHitTestSource({ space: viewerSpace }).then((source) => {\n this.source = source;\n });\n });\n\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.addEventListener('end', () => {\n this.sourceRequest = false;\n this.source = null;\n this.hand = null;\n });\n\n this.sourceRequest = true;\n }\n });\n\n this.app.addEventListener('exitxr', () => {\n this.deleteAnchor(this.app);\n this.app.origin.matrix.copy(new three__WEBPACK_IMPORTED_MODULE_4__.Matrix4());\n });\n\n document.addEventListener('selectstart', (event) => {\n if (this.currentEntity == null || (this.hand && this.hand != event.detail.handedness)) {\n return;\n }\n if (!event.detail) {\n return;\n }\n this.hand = event.detail.handedness;\n });\n\n document.addEventListener('selectmoved', (event) => {\n if (this.currentEntity && this.hand == event.detail.handedness) {\n this.userWorldPosition.setFromMatrixPosition(this.app.camera.matrixWorld);\n this.cameraForward.setFromMatrixPosition(this.app.user.forward.matrixWorld);\n\n this.pinchDistance = this.cameraForward.distanceTo(event.detail.position);\n this.scale = Math.exp(2 * this.pinchDistance);\n this.app.anchor.position.z = this.app.user.forward.position.z * this.scale;\n this.app.anchor.lookAt(this.userWorldPosition);\n }\n });\n\n document.addEventListener('selectend', (event) => {\n if (this.currentEntity == null || this.hand == null || this.hand != event.detail.handedness) {\n return;\n }\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n frame.createAnchor(this.matrix4ToXRRigidTransform(this.currentEntity.object3D.matrixWorld), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n this.currentEntity.anchor = anchor;\n this.anchoringQueue.delete(this.currentEntity);\n this.currentEntity = null;\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n this.hand = null;\n });\n }\n\n /**\n * @function\n * @description This update function maintains the transforms of anchored entities.\n * This overrides any other transform values set on the element when in mixed reality.\n * @param {number} deltaTime - given timestep to be used for any feature changes\n * @param {object} frame - given frame information to be used for any feature changes\n */\n update(deltaTime, frame) {\n if (mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.isPresenting) {\n if (this.currentEntity) {\n this.floating(frame);\n }\n\n if (!this.app.anchor) {\n this.setAppOrigin();\n } else {\n this.updateOrigin(frame);\n }\n }\n\n for (const entity of this.registry) {\n if (mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.isPresenting) {\n let anchorComp = entity.components.get('anchor');\n if (entity.anchor == null && !this.anchoringQueue.has(entity)) {\n entity.object3D.matrixWorldAutoUpdate = false;\n this.createAnchor(entity, anchorComp);\n } else if (entity.anchor) {\n let pose = frame.getPose(entity.anchor.anchorSpace, mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace);\n let transform = this.multiplyQuaternionWithXRRigidTransform(this.axisSwapQuat, pose.transform);\n\n entity.object3D.matrix.copy(this.adjustTransform(transform));\n } else {\n this.createAnchor(entity, anchorComp);\n }\n } else if (entity.anchor) {\n entity.object3D.matrix.copy(entity.object3D.userData.originalMatrix);\n this.deleteAnchor(entity);\n }\n }\n }\n\n /**\n * @function\n * @description Called when the entity component is initialized\n * @param {object} entity - the entity being attached/initialized.\n */\n attachedComponent(entity) {\n entity.object3D.userData.originalMatrix = new three__WEBPACK_IMPORTED_MODULE_4__.Matrix4();\n entity.object3D.userData.originalMatrix.copy(entity.object3D.matrixWorld);\n entity.object3D.matrixAutoUpdate = false;\n }\n\n /**\n * @function\n * @description Called when the entity component is updated\n * @param {object} entity - the entity being updated based on the component.\n */\n updatedComponent(entity) {\n // delete before creating a new one.\n this.deleteAnchor(entity);\n // // These cases can be managed instantly\n // this.createAnchor(entity, comp);\n }\n\n /**\n * @function\n * @description Called when the entity component is detached\n * @param {object} entity - the entity being updated based on the component being detached.\n */\n detachedComponent(entity) {\n entity.object3D.matrixAutoUpdate = true;\n this.deleteAnchor(entity);\n }\n\n /**\n * @function\n * @description deletes anchors from the scene and removes all references to the anchored plane (if any)\n * @param {object} entity - the entity whose anchor is being deleted.\n */\n deleteAnchor(entity) {\n if (entity.plane) {\n entity.plane.occupied = false;\n entity.plane.mesh.visible = true;\n entity.plane = null;\n }\n entity.anchor = null;\n entity.dispatchEvent(new CustomEvent('anchorremoved', { bubbles: true }));\n }\n\n /**\n * @function\n * @description creates the anchor specified by the data-anchor-comp\n * @param {object} entity - the entity whose anchor is being created.\n * @param {object} comp - the data component with a type value that represents the string 'fixed', 'plane', 'floating', etc\n */\n createAnchor(entity, comp) {\n switch (comp.type) {\n case 'fixed':\n this.fixed(entity);\n break;\n case 'plane':\n this.plane(entity, comp);\n break;\n case 'floating':\n this.currentEntity = entity;\n this.anchoringQueue.add(entity);\n break;\n default:\n break;\n }\n }\n\n /**\n * @function\n * @description Sets the origin of the MRApp being touched by all systems to allow anchoring to position\n * itself properly.\n */\n setAppOrigin() {\n if (!mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.isPresenting) {\n return;\n }\n let originMatrix = new three__WEBPACK_IMPORTED_MODULE_4__.Matrix4();\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n originMatrix.copyPosition(this.app.user.forward.matrixWorld);\n frame.createAnchor(this.matrix4ToXRRigidTransform(originMatrix), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n this.app.origin.matrixAutoUpdate = false;\n this.app.anchor = anchor;\n this.app.dispatchEvent(new CustomEvent('anchored', { bubbles: true }));\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n }\n\n /**\n * @function\n * @description Updates the origin of the MRApp being touched by all systems to allow anchoring to position.\n * @param {object} frame - given frame information to be used for any feature changes (from the update(..) loop)\n */\n updateOrigin(frame) {\n let pose = frame.getPose(this.app.anchor.anchorSpace, mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace);\n let transform = this.multiplyQuaternionWithXRRigidTransform(this.axisSwapQuat, pose.transform);\n\n this.app.origin.matrix.copy(this.adjustTransform(transform, true));\n\n this.originPosition.setFromMatrixPosition(this.app.origin.matrixWorld);\n }\n\n /**\n * @function\n * @description Anchors the given entity half a meter in front of the users position at launch.\n * @param {object} entity - the entity being positioned.\n */\n fixed(entity) {\n this.anchoringQueue.add(entity);\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n frame.createAnchor(this.matrix4ToXRRigidTransform(this.app.user.forward.matrixWorld), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n entity.anchor = anchor;\n entity.dispatchEvent(new CustomEvent('anchored', { bubbles: true }));\n this.anchoringQueue.delete(entity);\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n }\n\n /**\n * @function\n * @description Creates an anchor at the position specified by the user,\n * either floating in front of them or pinned to the scene mesh\n * @param {object} frame - given frame information to be used for any feature changes (from the update(..) loop)\n */\n floating(frame) {\n this.hitResults = frame.getHitTestResults(this.source);\n const hit = this.hitResults[0];\n const pose = hit?.getPose(mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace);\n\n if (pose && this.userWorldPosition.distanceTo(pose.transform.position) < this.snapDistance * this.scale) {\n let transform = this.multiplyQuaternionWithXRRigidTransform(this.axisSwapQuat, pose.transform);\n this.currentEntity.object3D.matrix.copy(this.adjustTransform(transform));\n } else {\n this.currentEntity.object3D.matrix.copy(this.app.anchor.matrixWorld);\n }\n }\n\n /**\n * @function\n * @description Anchors the provided entity to the nearest unoccupied plane that meets the given orientation and label.\n * each plane is currently limited to one anchor for simplicity.\n * @param {object} entity - the entity being anchored by this function.\n * @param {object} comp - the data-component to determine the orientation and label of the associated plane\n */\n plane(entity, comp) {\n this.anchoringQueue.add(entity);\n this.userWorldPosition.setFromMatrixPosition(this.app.user.forward.matrixWorld);\n let sort = Array.from(this.planeManager.planeDictionary[comp.label].values());\n sort.sort((a, b) => {\n return a.mesh.position.distanceTo(this.userWorldPosition) - b.mesh.position.distanceTo(this.userWorldPosition);\n });\n for (const mrPlane of sort) {\n if (mrPlane.occupied) {\n continue;\n }\n if (comp.label && comp.label != mrPlane.label) {\n continue;\n }\n if (comp.orientation && comp.orientation != mrPlane.orientation) {\n continue;\n }\n mrPlane.occupied = true;\n\n mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.session.requestAnimationFrame((t, frame) => {\n frame.createAnchor(this.matrix4ToXRRigidTransform(mrPlane.mesh.matrixWorld), mrjs__WEBPACK_IMPORTED_MODULE_0__.mrjsUtils.xr.referenceSpace).then(\n (anchor) => {\n this.anchoringQueue.delete(entity);\n if (!this.planeManager.planeDictionary[comp.label].has(mrPlane)) {\n return;\n }\n if (entity.anchor) {\n return;\n }\n entity.anchor = anchor;\n entity.plane = mrPlane;\n entity.dispatchEvent(new CustomEvent('anchored', { bubbles: true }));\n if (comp.occlusion == false) {\n mrPlane.mesh.visible = false;\n }\n },\n (error) => {\n console.error('Could not create anchor: ' + error);\n }\n );\n });\n return;\n }\n }\n\n originalAnchorMatrix = new three__WEBPACK_IMPORTED_MODULE_4__.Matrix4();\n anchorForwardVector = new three__WEBPACK_IMPORTED_MODULE_4__.Vector3(0, 0, 0.03);\n rotationMatrix = new three__WEBPACK_IMPORTED_MODULE_4__.Matrix4();\n adjustedMatrix = new three__WEBPACK_IMPORTED_MODULE_4__.Matrix4();\n\n /**\n * @function\n * @description converts the provided XRRigidTransform to a Matrix4 and adjusts it to ensure\n * that it's y-axis is pointing directly up and it's z-axis is facing inward\n * @param {object} xrRigidTransform - a THREE.js transformation matrix that we want to adjust\n * @param {boolean} origin - true if this is positioned at the origin or not (handles special case of div-0).\n * @returns {object} a new adjusted THREE.js Matrix4\n */\n adjustTransform(xrRigidTransform, origin = false) {\n // Create a Three.js Quaternion for the XRRigidTransform's orientation\n let quaternion = new three__WEBPACK_IMPORTED_MODULE_4__.Quaternion(xrRigidTransform.orientation.x, xrRigidTransform.orientation.y, xrRigidTransform.orientation.z, xrRigidTransform.orientation.w);\n\n // Create a Three.js Vector for the up direction\n let upVector = new three__WEBPACK_IMPORTED_MODULE_4__.Vector3(0, 1, 0);\n\n // Apply the quaternion to the up direction\n let transformedUp = upVector.clone().applyQuaternion(quaternion);\n\n // Calculate the rotation needed to align the transformed up direction with the global up direction\n let correctionQuaternion = new three__WEBPACK_IMPORTED_MODULE_4__.Quaternion().setFromUnitVectors(transformedUp, upVector);\n\n // Apply the correction to the original quaternion\n quaternion.premultiply(correctionQuaternion);\n\n // Create a new Three.js Vector3 for the position\n let position = new three__WEBPACK_IMPORTED_MODULE_4__.Vector3(xrRigidTransform.position.x, xrRigidTransform.position.y, xrRigidTransform.position.z);\n\n if (!origin) {\n position.sub(this.originPosition);\n }\n\n this.originalAnchorMatrix.compose(position, quaternion, new three__WEBPACK_IMPORTED_MODULE_4__.Vector3(1, 1, 1));\n this.rotationMatrix.extractRotation(this.originalAnchorMatrix);\n\n this.anchorForwardVector.applyMatrix4(this.rotationMatrix);\n this.adjustedMatrix.makeTranslation(this.anchorForwardVector.x, this.anchorForwardVector.y, this.anchorForwardVector.y);\n this.anchorForwardVector.set(0, 0, 0.03);\n this.adjustedMatrix.multiply(this.originalAnchorMatrix);\n\n return this.adjustedMatrix;\n }\n\n /**\n * @function\n * @description Converts the provided matrix4 into a webXR xompatible XRRigidTransform\n * @param {object} matrix4 - the matrix we want to convert to a XRRigidTransform\n * @returns {object} xrRigidTransform - the converted representation of the param matrix4\n */\n matrix4ToXRRigidTransform(matrix4) {\n // Extract the translation component from the Matrix4\n const position = new three__WEBPACK_IMPORTED_MODULE_4__.Vector3();\n position.setFromMatrixPosition(matrix4);\n\n // Extract the rotation component from the Matrix4\n const quaternion = new three__WEBPACK_IMPORTED_MODULE_4__.Quaternion();\n quaternion.setFromRotationMatrix(matrix4);\n\n // Create a DOMPointInit for the position\n const positionInit = { x: position.x, y: position.y, z: position.z, w: 1 };\n\n // Create a DOMPointInit for the orientation (quaternion)\n const orientationInit = { x: quaternion.x, y: quaternion.y, z: quaternion.z, w: quaternion.w };\n\n // Create and return the XRRigidTransform\n return new XRRigidTransform(positionInit, orientationInit);\n }\n\n /**\n * @function\n * @description Multiplies an xr rigid transform by the provided quaternion\n * @param {object} quaternion - the quaternion we want to multiply with the xrRigidTransform\n * @param {object} xrRigidTransform - the second part of the multiplication we are looking to perform.\n * @returns {object} xrRigidTransform - the output of the quaternion * xrRigidTransform in the form of an xrRigidTransform\n */\n multiplyQuaternionWithXRRigidTransform(quaternion, xrRigidTransform) {\n // Create a Three.js Quaternion for the XRRigidTransform's orientation\n let transformQuaternion = new three__WEBPACK_IMPORTED_MODULE_4__.Quaternion(\n xrRigidTransform.orientation.x,\n xrRigidTransform.orientation.y,\n xrRigidTransform.orientation.z,\n xrRigidTransform.orientation.w\n );\n\n // Multiply the quaternions\n transformQuaternion.multiply(quaternion);\n\n // Create a new XRRigidTransform with the multiplied orientation and original position\n let newPosition = xrRigidTransform.position;\n let newOrientation = new DOMPointReadOnly(transformQuaternion.x, transformQuaternion.y, transformQuaternion.z, transformQuaternion.w);\n\n return new XRRigidTransform(newPosition, newOrientation);\n }\n}\n\n\n//# sourceURL=webpack://mrjs/./src/core/componentSystems/AnchorSystem.js?"); /***/ }), @@ -696,7 +696,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRImageEntity: () => (/* binding */ MRImageEntity)\n/* harmony export */ });\n/* harmony import */ var mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRMediaEntity */ \"./src/core/entities/MRMediaEntity.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n\n\n\n\n\n\n/**\n * @class MRImageEntity\n * @classdesc Base html image represented in 3D space. `mr-image`\n * @augments MRMediaEntity\n */\nclass MRImageEntity extends mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__.MRMediaEntity {\n /**\n * @class\n * @description Constructs a base image entity using a UIPlane and other 3D elements as necessary.\n */\n constructor() {\n super();\n\n // object3D and rest of mrvideo is pre-created in MRMediaEntity\n this.object3D.name = 'image';\n }\n\n /**\n * @function\n * @description Gets the width of the internal media object\n * @returns {number} width - the value of the width\n */\n get mediaWidth() {\n return this.media.width;\n }\n\n /**\n * @function\n * @description Gets the height of the internal media object\n * @returns {number} height - the value of the height\n */\n get mediaHeight() {\n return this.media.height;\n }\n\n /**\n * @function\n * @description (async) handles setting up this Image and associated 3D geometry style (from css) once it is connected to run as an entity component.\n */\n async connected() {\n await super.connected();\n this.media = document.createElement('img');\n super.connected();\n }\n\n /**\n * @function\n * @description Loads the Media texture of the setup this.media object based on its html source info.\n */\n loadMediaTexture() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.material\n .loadTextureAsync(this.media.src)\n .then((texture) => {\n this.texture = texture;\n this.object3D.material.map = texture;\n })\n .catch((error) => {\n console.error('Error loading texture:', error);\n });\n }\n\n /**\n * @function\n * @description Callback function of MREntity - Updates the image's cover,fill,etc based on the mutation request.\n * @param {object} mutation - the update/change/mutation to be handled.\n */\n mutated(mutation) {\n // Mutations are only understood by their actual type. Any mutation\n // passed through MRMediaEntity directly is undefined since it is not\n // a direct element for users. So we do the if-check here and then\n // follow the same as the parent's functionality.\n if (mutation.type == 'attributes' && mutation.attributeName == 'src') {\n super.mutated();\n }\n }\n}\n\ncustomElements.get('mr-img') || customElements.define('mr-img', MRImageEntity);\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRImageEntity.js?"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRImageEntity: () => (/* binding */ MRImageEntity)\n/* harmony export */ });\n/* harmony import */ var mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRMediaEntity */ \"./src/core/entities/MRMediaEntity.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n\n\n\n\n\n\n/**\n * @class MRImageEntity\n * @classdesc Base html image represented in 3D space. `mr-image`\n * @augments MRMediaEntity\n */\nclass MRImageEntity extends mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__.MRMediaEntity {\n /**\n * @class\n * @description Constructs a base image entity using a UIPlane and other 3D elements as necessary.\n */\n constructor() {\n super();\n\n // object3D and rest of mrvideo is pre-created in MRMediaEntity\n this.object3D.name = 'image';\n }\n\n /**\n * @function\n * @description Gets the width of the internal media object\n * @returns {number} width - the value of the width\n */\n get mediaWidth() {\n return this.media.width;\n }\n\n /**\n * @function\n * @description Gets the height of the internal media object\n * @returns {number} height - the value of the height\n */\n get mediaHeight() {\n return this.media.height;\n }\n\n /**\n * @function\n * @description (async) handles setting up this Image and associated 3D geometry style (from css) once it is connected to run as an entity component.\n */\n async connected() {\n this.media = document.createElement('img');\n await super.connected();\n }\n\n /**\n * @function\n * @description Loads the Media texture of the setup this.media object based on its html source info.\n */\n loadMediaTexture() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.material\n .loadTextureAsync(this.media.src)\n .then((texture) => {\n this.texture = texture;\n this.object3D.material.map = texture;\n })\n .catch((error) => {\n console.error('Error loading texture:', error);\n });\n }\n\n /**\n * @function\n * @description Callback function of MREntity - Updates the image's cover,fill,etc based on the mutation request.\n * @param {object} mutation - the update/change/mutation to be handled.\n */\n mutated(mutation) {\n // Mutations are only understood by their actual type. Any mutation\n // passed through MRMediaEntity directly is undefined since it is not\n // a direct element for users. So we do the if-check here and then\n // follow the same as the parent's functionality.\n if (mutation.type == 'attributes' && mutation.attributeName == 'src') {\n super.mutated();\n }\n }\n}\n\ncustomElements.get('mr-img') || customElements.define('mr-img', MRImageEntity);\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRImageEntity.js?"); /***/ }), @@ -718,7 +718,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRMediaEntity: () => (/* binding */ MRMediaEntity)\n/* harmony export */ });\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs_core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRDivEntity */ \"./src/core/entities/MRDivEntity.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n\n\n\n\n\n\n/**\n * @class MRMediaEntity\n * @classdesc Base html media entity represented in 3D space. `mr-media`\n * @augments MRDivEntity\n */\nclass MRMediaEntity extends mrjs_core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_0__.MRDivEntity {\n /**\n * @class\n * @description Constructs a base media entity using a UIPlane and other 3D elements as necessary.\n */\n constructor() {\n super();\n this.attachShadow({ mode: 'open' });\n\n // Create the object3D. Dont need default value for geometry\n // until the connected call since this will get overwritten anyways.\n const material = new three__WEBPACK_IMPORTED_MODULE_2__.MeshStandardMaterial({\n side: three__WEBPACK_IMPORTED_MODULE_2__.FrontSide,\n });\n // Object3D for MRMediaEntity (mrimage,mrvideo,etc) is the actual image/video/etc itself in 3D space\n this.object3D = new three__WEBPACK_IMPORTED_MODULE_2__.Mesh(undefined, material);\n this.object3D.receiveShadow = true;\n this.object3D.renderOrder = 3;\n this.object3D.name = 'media';\n\n // the media to be filled out.\n // for ex: document.createElement('video') or document.createElement('img');\n this.media = null;\n\n // This is a reference to the texture that is used as part of the\n // threejs material. Separating it out for easier updating after it is loaded.\n // The texture is filled-in in the connected function.\n this.texture = null;\n\n // This is used to aid in the formatting for certain object-fit setups\n // ex: contain, scale-down\n this.subMediaMesh = new three__WEBPACK_IMPORTED_MODULE_2__.Mesh();\n this.subMediaMesh.receiveShadow = true;\n this.subMediaMesh.renderOrder = 3;\n this.subMediaMesh.name = 'subMedia';\n this.object3D.add(this.subMediaMesh);\n }\n\n /**\n * @function\n * @description Calculates the width of the MRMedia object\n * @returns {number} - the resolved width\n */\n get width() {\n let width = this.objectFitDimensions?.width;\n return width > 0 ? width : super.width;\n }\n\n /**\n * @function\n * @description Calculates the height of the MRMedia object\n * @returns {number} - the resolved height\n */\n get height() {\n let height = this.objectFitDimensions?.height;\n return height > 0 ? height : super.height;\n }\n\n /**\n * @function\n * @description Calculates the width of the media based on the media tag itself\n * This function will error if called directly as an MRMedia item. Made to be overridden\n * by children.\n * @returns {number} - the resolved height\n */\n get mediaWidth() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.error.emptyParentFunction();\n return null;\n }\n\n /**\n * @function\n * @description Calculates the height of the media based on the media tag itself\n * This function will error if called directly as an MRMedia item. Made to be overridden\n * by children.\n * @returns {number} - the resolved height\n */\n get mediaHeight() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.error.emptyParentFunction();\n return null;\n }\n\n /**\n * @function\n * @description Creates the Media Plane Geometry used to draw the Image,Video,etc\n * This is a separate object to allow for common css styling such as 'contain' and 'scale-down'.\n */\n generateNewMediaPlaneGeometry() {\n if (this.object3D.geometry !== undefined) {\n this.object3D.geometry.dispose();\n }\n this.object3D.geometry = mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.geometry.UIPlane(this.width, this.height, this.borderRadii, 18);\n }\n\n /**\n * @function\n * @description Loads the associated media into 3D based on its html properties.\n * This function will error if called directly as an MRMedia item. Made to be overridden\n * by children.\n */\n loadMediaTexture() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description (async) handles setting up this media and associated 3D geometry style (from css) once it is connected to run as an entity component.\n */\n async connected() {\n this.media.setAttribute('src', mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.html.resolvePath(this.getAttribute('src')));\n this.media.setAttribute('style', 'object-fit:inherit; width:inherit');\n\n this.objectFitDimensions = { height: 0, width: 0 };\n if (this.getAttribute('src') !== undefined) {\n this.computeObjectFitDimensions();\n this.generateNewMediaPlaneGeometry();\n this.loadMediaTexture();\n }\n }\n\n /**\n * @function\n * @description Callback function of MREntity - Updates the media's cover,fill,etc based on the mutation request.\n * @param {object} mutation - the update/change/mutation to be handled.\n */\n mutated(mutation) {\n super.mutated();\n\n if (mutation.type == 'attributes' && mutation.attributeName == 'src') {\n this.media.setAttribute('src', this.getAttribute('src'));\n this.computeObjectFitDimensions();\n this.loadMediaTexture();\n }\n }\n\n /**\n * @function\n * @description computes the width and height values for the image considering the value of object-fit\n */\n computeObjectFitDimensions() {\n if (!this.texture || !this.media) {\n // We assume every media item exists and has its attached texture.\n // If texture doesnt exist, it's just not loaded in yet, meaning\n // we can skip the below until it is.\n return;\n }\n\n const _oldSubMediaMeshNotNeeded = () => {\n // All switch options other than 'contain' and 'scale-down' start with this\n // function.\n //\n // 'contain' and 'scale-down' are the only options that use this additional\n // mesh for positioning.\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.model.disposeObject3D(this.subMediaMesh);\n };\n\n const _showMainMediaMesh = () => {\n // used if transitioning away from 'contain' or 'scale-down'\n // to make sure that texture is set properly\n this.object3D.material.visible = true;\n this.object3D.material.needsUpdate = true;\n };\n\n const _hideMainMediaMesh = () => {\n // to parallel the '_makeSureMainMediaMeshHasTexture' for readability\n // and debugging later on.\n this.object3D.material.visible = false;\n this.object3D.material.needsUpdate = true;\n };\n\n let containerWidth = this.parentElement.width;\n let containerHeight = this.parentElement.height;\n let mediaWidth = this.mediaWidth;\n let mediaHeight = this.mediaHeight;\n const mediaAspect = mediaWidth / mediaHeight;\n const containerAspect = containerWidth / containerHeight;\n switch (this.compStyle.objectFit) {\n case 'fill':\n _oldSubMediaMeshNotNeeded();\n _showMainMediaMesh();\n this.objectFitDimensions = { width: containerWidth, height: containerHeight };\n\n break;\n\n case 'contain':\n case 'scale-down':\n // `contain` and `scale-down` are the same except for one factor:\n // - `contain` will always scale the media to fit\n // - `scale-down` will only scale the media to fit if the media is larger than the container\n //\n // Implemented by making the main mesh turn into the same dimensions as the container\n // and making the submesh scale itself based on those values. This allows the original hit\n // box of mr-media to stay as expected while the media itself still changes based on object-fit.\n\n if (this.compStyle.objectFit === 'contain' || mediaWidth > containerWidth || mediaHeight > containerHeight) {\n const scaleRatio = Math.min(containerWidth / mediaWidth, containerHeight / mediaHeight);\n mediaWidth *= scaleRatio;\n mediaHeight *= scaleRatio;\n }\n\n const mediaGeometry = new three__WEBPACK_IMPORTED_MODULE_2__.PlaneGeometry(mediaWidth, mediaHeight);\n const mediaMaterial = new three__WEBPACK_IMPORTED_MODULE_2__.MeshStandardMaterial({\n map: this.texture,\n });\n _oldSubMediaMeshNotNeeded();\n this.subMediaMesh.geometry = mediaGeometry;\n this.subMediaMesh.material = mediaMaterial;\n this.subMediaMesh.geometry.needsUpdate = true;\n this.subMediaMesh.material.visible = true;\n this.subMediaMesh.material.needsUpdate = true;\n _hideMainMediaMesh();\n\n this.objectFitDimensions = {\n width: containerWidth,\n height: containerHeight,\n };\n\n break;\n\n case 'cover':\n _oldSubMediaMeshNotNeeded();\n _showMainMediaMesh();\n\n this.texture.repeat.set(1, 1); // Reset scaling\n\n if (containerAspect > mediaAspect) {\n // Plane is wider than the texture\n const scale = containerAspect / mediaAspect;\n this.texture.repeat.y = 1 / scale;\n this.texture.offset.y = (1 - 1 / scale) / 2; // Center the texture vertically\n } else {\n // Plane is taller than the texture\n const scale = mediaAspect / containerAspect;\n this.texture.repeat.x = 1 / scale;\n this.texture.offset.x = (1 - 1 / scale) / 2; // Center the texture horizontally\n }\n this.texture.needsUpdate = true; // Important to update the texture\n\n this.objectFitDimensions = {\n width: containerWidth,\n height: containerHeight,\n };\n\n break;\n\n case 'none':\n _oldSubMediaMeshNotNeeded();\n _showMainMediaMesh();\n this.objectFitDimensions = { width: mediaWidth, height: mediaHeight };\n\n break;\n\n default:\n throw new Error(`Unsupported object-fit value ${this.compStyle.objectFit}`);\n }\n\n this.style.width = `${this.objectFitDimensions.width}px`;\n this.style.height = `${this.objectFitDimensions.height}px`;\n }\n}\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRMediaEntity.js?"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRMediaEntity: () => (/* binding */ MRMediaEntity)\n/* harmony export */ });\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs_core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRDivEntity */ \"./src/core/entities/MRDivEntity.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n\n\n\n\n\n\n/**\n * @class MRMediaEntity\n * @classdesc Base html media entity represented in 3D space. `mr-media`\n * @augments MRDivEntity\n */\nclass MRMediaEntity extends mrjs_core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_0__.MRDivEntity {\n /**\n * @class\n * @description Constructs a base media entity using a UIPlane and other 3D elements as necessary.\n */\n constructor() {\n super();\n this.attachShadow({ mode: 'open' });\n\n // Create the object3D. Dont need default value for geometry\n // until the connected call since this will get overwritten anyways.\n const material = new three__WEBPACK_IMPORTED_MODULE_2__.MeshStandardMaterial({\n side: three__WEBPACK_IMPORTED_MODULE_2__.FrontSide,\n });\n // Object3D for MRMediaEntity (mrimage,mrvideo,etc) is the actual image/video/etc itself in 3D space\n this.object3D = new three__WEBPACK_IMPORTED_MODULE_2__.Mesh(undefined, material);\n this.object3D.receiveShadow = true;\n this.object3D.renderOrder = 3;\n this.object3D.name = 'media';\n\n // the media to be filled out.\n // for ex: document.createElement('video') or document.createElement('img');\n this.media = null;\n\n // This is a reference to the texture that is used as part of the\n // threejs material. Separating it out for easier updating after it is loaded.\n // The texture is filled-in in the connected function.\n this.texture = null;\n\n // This is used to aid in the formatting for certain object-fit setups\n // ex: contain, scale-down\n this.subMediaMesh = new three__WEBPACK_IMPORTED_MODULE_2__.Mesh();\n this.subMediaMesh.receiveShadow = true;\n this.subMediaMesh.renderOrder = 3;\n this.subMediaMesh.name = 'subMedia';\n this.object3D.add(this.subMediaMesh);\n }\n\n /**\n * @function\n * @description Calculates the width of the MRMedia object\n * @returns {number} - the resolved width\n */\n get width() {\n let width = this.objectFitDimensions?.width;\n return width > 0 ? width : super.width;\n }\n\n /**\n * @function\n * @description Calculates the height of the MRMedia object\n * @returns {number} - the resolved height\n */\n get height() {\n let height = this.objectFitDimensions?.height;\n return height > 0 ? height : super.height;\n }\n\n /**\n * @function\n * @description Calculates the width of the media based on the media tag itself\n * This function will error if called directly as an MRMedia item. Made to be overridden\n * by children.\n * @returns {number} - the resolved height\n */\n get mediaWidth() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.error.emptyParentFunction();\n return null;\n }\n\n /**\n * @function\n * @description Calculates the height of the media based on the media tag itself\n * This function will error if called directly as an MRMedia item. Made to be overridden\n * by children.\n * @returns {number} - the resolved height\n */\n get mediaHeight() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.error.emptyParentFunction();\n return null;\n }\n\n /**\n * @function\n * @description Creates the Media Plane Geometry used to draw the Image,Video,etc\n * This is a separate object to allow for common css styling such as 'contain' and 'scale-down'.\n */\n generateNewMediaPlaneGeometry() {\n if (this.object3D.geometry !== undefined) {\n this.object3D.geometry.dispose();\n }\n this.object3D.geometry = mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.geometry.UIPlane(this.width, this.height, this.borderRadii, 18);\n }\n\n /**\n * @function\n * @description Loads the associated media into 3D based on its html properties.\n * This function will error if called directly as an MRMedia item. Made to be overridden\n * by children.\n */\n loadMediaTexture() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description (async) handles setting up this media and associated 3D geometry style (from css) once it is connected to run as an entity component.\n */\n async connected() {\n await super.connected();\n this.media.setAttribute('src', mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.html.resolvePath(this.getAttribute('src')));\n this.media.setAttribute('style', 'object-fit:inherit; width:inherit');\n\n this.objectFitDimensions = { height: 0, width: 0 };\n if (this.getAttribute('src') !== undefined) {\n this.computeObjectFitDimensions();\n this.generateNewMediaPlaneGeometry();\n this.loadMediaTexture();\n }\n }\n\n /**\n * @function\n * @description Callback function of MREntity - Updates the media's cover,fill,etc based on the mutation request.\n * @param {object} mutation - the update/change/mutation to be handled.\n */\n mutated(mutation) {\n super.mutated();\n\n if (mutation.type == 'attributes' && mutation.attributeName == 'src') {\n this.media.setAttribute('src', this.getAttribute('src'));\n this.computeObjectFitDimensions();\n this.loadMediaTexture();\n }\n }\n\n /**\n * @function\n * @description computes the width and height values for the image considering the value of object-fit\n */\n computeObjectFitDimensions() {\n if (!this.texture || !this.media) {\n // We assume every media item exists and has its attached texture.\n // If texture doesnt exist, it's just not loaded in yet, meaning\n // we can skip the below until it is.\n return;\n }\n\n const _oldSubMediaMeshNotNeeded = () => {\n // All switch options other than 'contain' and 'scale-down' start with this\n // function.\n //\n // 'contain' and 'scale-down' are the only options that use this additional\n // mesh for positioning.\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.model.disposeObject3D(this.subMediaMesh);\n };\n\n const _showMainMediaMesh = () => {\n // used if transitioning away from 'contain' or 'scale-down'\n // to make sure that texture is set properly\n this.object3D.material.visible = true;\n this.object3D.material.needsUpdate = true;\n };\n\n const _hideMainMediaMesh = () => {\n // to parallel the '_makeSureMainMediaMeshHasTexture' for readability\n // and debugging later on.\n this.object3D.material.visible = false;\n this.object3D.material.needsUpdate = true;\n };\n\n let containerWidth = this.parentElement.width;\n let containerHeight = this.parentElement.height;\n let mediaWidth = this.mediaWidth;\n let mediaHeight = this.mediaHeight;\n const mediaAspect = mediaWidth / mediaHeight;\n const containerAspect = containerWidth / containerHeight;\n switch (this.compStyle.objectFit) {\n case 'fill':\n _oldSubMediaMeshNotNeeded();\n _showMainMediaMesh();\n this.objectFitDimensions = { width: containerWidth, height: containerHeight };\n\n break;\n\n case 'contain':\n case 'scale-down':\n // `contain` and `scale-down` are the same except for one factor:\n // - `contain` will always scale the media to fit\n // - `scale-down` will only scale the media to fit if the media is larger than the container\n //\n // Implemented by making the main mesh turn into the same dimensions as the container\n // and making the submesh scale itself based on those values. This allows the original hit\n // box of mr-media to stay as expected while the media itself still changes based on object-fit.\n\n if (this.compStyle.objectFit === 'contain' || mediaWidth > containerWidth || mediaHeight > containerHeight) {\n const scaleRatio = Math.min(containerWidth / mediaWidth, containerHeight / mediaHeight);\n mediaWidth *= scaleRatio;\n mediaHeight *= scaleRatio;\n }\n\n const mediaGeometry = new three__WEBPACK_IMPORTED_MODULE_2__.PlaneGeometry(mediaWidth, mediaHeight);\n const mediaMaterial = new three__WEBPACK_IMPORTED_MODULE_2__.MeshStandardMaterial({\n map: this.texture,\n });\n _oldSubMediaMeshNotNeeded();\n this.subMediaMesh.geometry = mediaGeometry;\n this.subMediaMesh.material = mediaMaterial;\n this.subMediaMesh.geometry.needsUpdate = true;\n this.subMediaMesh.material.visible = true;\n this.subMediaMesh.material.needsUpdate = true;\n _hideMainMediaMesh();\n\n this.objectFitDimensions = {\n width: containerWidth,\n height: containerHeight,\n };\n\n break;\n\n case 'cover':\n _oldSubMediaMeshNotNeeded();\n _showMainMediaMesh();\n\n this.texture.repeat.set(1, 1); // Reset scaling\n\n if (containerAspect > mediaAspect) {\n // Plane is wider than the texture\n const scale = containerAspect / mediaAspect;\n this.texture.repeat.y = 1 / scale;\n this.texture.offset.y = (1 - 1 / scale) / 2; // Center the texture vertically\n } else {\n // Plane is taller than the texture\n const scale = mediaAspect / containerAspect;\n this.texture.repeat.x = 1 / scale;\n this.texture.offset.x = (1 - 1 / scale) / 2; // Center the texture horizontally\n }\n this.texture.needsUpdate = true; // Important to update the texture\n\n this.objectFitDimensions = {\n width: containerWidth,\n height: containerHeight,\n };\n\n break;\n\n case 'none':\n _oldSubMediaMeshNotNeeded();\n _showMainMediaMesh();\n this.objectFitDimensions = { width: mediaWidth, height: mediaHeight };\n\n break;\n\n default:\n throw new Error(`Unsupported object-fit value ${this.compStyle.objectFit}`);\n }\n\n this.style.width = `${this.objectFitDimensions.width}px`;\n this.style.height = `${this.objectFitDimensions.height}px`;\n }\n}\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRMediaEntity.js?"); /***/ }), @@ -751,7 +751,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRSkyBoxEntity: () => (/* binding */ MRSkyBoxEntity)\n/* harmony export */ });\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs_core_MREntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/MREntity */ \"./src/core/MREntity.js\");\n\n\n\n\n/**\n * @class MRSkyBoxEntity\n * @classdesc The skybox entity that allows users to give multiple images to pattern into the 3D background space. `mr-skybox`\n * @augments MREntity\n */\nclass MRSkyBoxEntity extends mrjs_core_MREntity__WEBPACK_IMPORTED_MODULE_0__.MREntity {\n /**\n * @class\n * @description Constructor for skybox - defaults to the usual impl of an Entity.\n */\n constructor() {\n super();\n this.object3D.name = 'skybox';\n\n this.skybox = null;\n this.textureLoadedCallbacks = [];\n }\n\n /**\n * @function\n * @description Callback function triggered when the texture is successfully loaded.\n * It sets the loaded texture as the background and notifies all registered callbacks.\n * @param {THREE.Texture} texture - The loaded texture.\n */\n onTextureLoaded(texture) {\n if (this.skybox) {\n if (Array.isArray(texture.images) && texture.images.length === 6) {\n // Handle cube texture case\n if (this.skybox.material !== undefined) {\n this.skybox.material.dispose();\n }\n this.skybox.material = new three__WEBPACK_IMPORTED_MODULE_1__.MeshStandardMaterial({\n envMap: texture,\n side: three__WEBPACK_IMPORTED_MODULE_1__.BackSide, // Render only on the inside\n });\n } else {\n // Handle single texture case\n if (this.skybox.material !== undefined) {\n this.skybox.material.dispose();\n }\n this.skybox.material = new three__WEBPACK_IMPORTED_MODULE_1__.MeshBasicMaterial({\n map: texture,\n side: three__WEBPACK_IMPORTED_MODULE_1__.BackSide, // Render only on the inside\n opacity: 1,\n });\n }\n }\n this.textureLoadedCallbacks.forEach((callback) => callback(texture));\n }\n\n /**\n * @function\n * @description (async) Lifecycle method that is called when the entity is connected.\n * This method initializes and starts the texture loading process.\n */\n async connected() {\n await super.connected();\n // you can have texturesList be all individual textures\n // or you can store them in a specified path and just\n // load them up solely by filename in that path.\n\n this.texturesList = mrjsUtils.html.resolvePath(this.getAttribute('src'));\n if (!this.texturesList) {\n return;\n }\n\n const textureNames = this.texturesList.split(',');\n const path = this.getAttribute('pathToTextures');\n const textureUrls = textureNames.map((name) => mrjsUtils.html.resolvePath(path ? path + name : name));\n\n let geometry;\n let textureLoader;\n if (textureNames.length > 1) {\n geometry = new three__WEBPACK_IMPORTED_MODULE_1__.BoxGeometry(900, 900, 900);\n textureLoader = new three__WEBPACK_IMPORTED_MODULE_1__.CubeTextureLoader();\n textureLoader.load(textureUrls, this.onTextureLoaded.bind(this));\n } else if (textureUrls.length == 1) {\n geometry = new three__WEBPACK_IMPORTED_MODULE_1__.SphereGeometry(900, 32, 16);\n textureLoader = new three__WEBPACK_IMPORTED_MODULE_1__.TextureLoader();\n textureLoader.load(textureUrls[0], this.onTextureLoaded.bind(this));\n }\n\n if (this.skybox) {\n // Remove existing skybox if present\n this.object3D.remove(this.skybox);\n this.skybox.dispose();\n }\n this.skybox = new three__WEBPACK_IMPORTED_MODULE_1__.Mesh(geometry); // going to passively load texture on async\n this.object3D.add(this.skybox);\n }\n\n /**\n * @function\n * @description Set the opacity of the skybox itself. Useful for blending between the outside and MR. Also\n * useful for cases where you want to blend between different skybox versions.\n */\n set setOpacity(val) {\n this.object3D.traverse((child) => {\n if (child.isMesh) {\n child.material.transparent = true;\n child.material.opacity = val;\n child.material.needsUpdate = true;\n }\n });\n }\n\n /**\n * @function\n * @description On load event function - right now defaults to do nothing.\n */\n onLoad = () => {};\n}\ncustomElements.define('mr-skybox', MRSkyBoxEntity);\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRSkyBoxEntity.js?"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRSkyBoxEntity: () => (/* binding */ MRSkyBoxEntity)\n/* harmony export */ });\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs_core_MREntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/MREntity */ \"./src/core/MREntity.js\");\n/* harmony import */ var mrjs_core_MRSystem__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs/core/MRSystem */ \"./src/core/MRSystem.js\");\n\n\n\n\n\n/**\n * @class MRSkyBoxEntity\n * @classdesc The skybox entity that allows users to give multiple images to pattern into the 3D background space. `mr-skybox`\n * @augments MREntity\n */\nclass MRSkyBoxEntity extends mrjs_core_MREntity__WEBPACK_IMPORTED_MODULE_0__.MREntity {\n /**\n * @class\n * @description Constructor for skybox - defaults to the usual impl of an Entity.\n */\n constructor() {\n super();\n this.object3D.name = 'skybox';\n\n this.skybox = null;\n this.textureLoadedCallbacks = [];\n }\n\n /**\n * @function\n * @description Callback function triggered when the texture is successfully loaded.\n * It sets the loaded texture as the background and notifies all registered callbacks.\n * @param {THREE.Texture} texture - The loaded texture.\n */\n onTextureLoaded(texture) {\n if (this.skybox) {\n if (Array.isArray(texture.images) && texture.images.length === 6) {\n // Handle cube texture case\n if (this.skybox.material !== undefined) {\n this.skybox.material.dispose();\n }\n this.skybox.material = new three__WEBPACK_IMPORTED_MODULE_2__.MeshStandardMaterial({\n envMap: texture,\n side: three__WEBPACK_IMPORTED_MODULE_2__.BackSide, // Render only on the inside\n });\n } else {\n // Handle single texture case\n if (this.skybox.material !== undefined) {\n this.skybox.material.dispose();\n }\n this.skybox.material = new three__WEBPACK_IMPORTED_MODULE_2__.MeshBasicMaterial({\n map: texture,\n side: three__WEBPACK_IMPORTED_MODULE_2__.BackSide, // Render only on the inside\n opacity: 1,\n });\n }\n }\n this.textureLoadedCallbacks.forEach((callback) => callback(texture));\n }\n\n /**\n * @function\n * @description (async) Lifecycle method that is called when the entity is connected.\n * This method initializes and starts the texture loading process.\n */\n async connected() {\n // you can have texturesList be all individual textures\n // or you can store them in a specified path and just\n // load them up solely by filename in that path.\n\n this.texturesList = mrjsUtils.html.resolvePath(this.getAttribute('src'));\n if (!this.texturesList) {\n return;\n }\n\n const textureNames = this.texturesList.split(',');\n const path = this.getAttribute('pathToTextures');\n const textureUrls = textureNames.map((name) => mrjsUtils.html.resolvePath(path ? path + name : name));\n\n let geometry;\n let textureLoader;\n if (textureNames.length > 1) {\n geometry = new three__WEBPACK_IMPORTED_MODULE_2__.BoxGeometry(900, 900, 900);\n textureLoader = new three__WEBPACK_IMPORTED_MODULE_2__.CubeTextureLoader();\n textureLoader.load(textureUrls, this.onTextureLoaded.bind(this));\n } else if (textureUrls.length == 1) {\n geometry = new three__WEBPACK_IMPORTED_MODULE_2__.SphereGeometry(900, 32, 16);\n textureLoader = new three__WEBPACK_IMPORTED_MODULE_2__.TextureLoader();\n textureLoader.load(textureUrls[0], this.onTextureLoaded.bind(this));\n }\n\n if (this.skybox) {\n // Remove existing skybox if present\n this.object3D.remove(this.skybox);\n this.skybox.dispose();\n }\n this.skybox = new three__WEBPACK_IMPORTED_MODULE_2__.Mesh(geometry); // going to passively load texture on async\n this.object3D.add(this.skybox);\n }\n\n /**\n * @function\n * @description Set the opacity of the skybox itself. Useful for blending between the outside and MR. Also\n * useful for cases where you want to blend between different skybox versions.\n */\n set setOpacity(val) {\n this.object3D.traverse((child) => {\n if (child.isMesh) {\n child.material.transparent = true;\n child.material.opacity = val;\n child.material.needsUpdate = true;\n }\n });\n }\n\n /**\n * @function\n * @description On load event function - right now defaults to do nothing.\n */\n onLoad = () => {};\n}\ncustomElements.define('mr-skybox', MRSkyBoxEntity);\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRSkyBoxEntity.js?"); /***/ }), @@ -806,7 +806,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRTextInputEntity: () => (/* binding */ MRTextInputEntity)\n/* harmony export */ });\n/* harmony import */ var troika_three_text__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! troika-three-text */ \"./node_modules/troika-three-text/dist/troika-three-text.esm.js\");\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs_core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRTextEntity */ \"./src/core/entities/MRTextEntity.js\");\n\n\n\n\n\n\n/**\n * @class MRTextInputEntity\n * @classdesc Base text inpu entity represented in 3D space. `mr-text-input`\n * @augments MRTextEntity\n */\nclass MRTextInputEntity extends mrjs_core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_0__.MRTextEntity {\n /**\n * @class\n * @description Constructor for the MRTextInputEntity entity component.\n */\n constructor() {\n super();\n this.attachShadow({ mode: 'open' });\n }\n\n /**\n * @function\n * @description Gets the value of the text for the current hiddenInput DOM object\n * @returns {string} value - the text value of the current hiddenInput DOM object\n */\n get value() {\n return this.hiddenInput.value;\n }\n\n /**\n * @function\n * @description Sets the value of the text for the current hiddenInput DOM object\n */\n set value(val) {\n this.hiddenInput.value = val;\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Called by connected to make sure\n * the hiddenInput dom element is created as expected.\n */\n createHiddenInputElement() {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Called by connected after\n * createHiddenInputElement to fill it in with the user's given\n * attribute information.\n */\n fillInHiddenInputElementWithUserData() {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Used on event trigger to\n * update the textObj visual based on the hiddenInput DOM element.\n * @param {boolean} fromCursorMove - default set as false if not supplied. See `MRTextArea`\n * and `MRTextField` as examples. This param is helpful for cases where the visible\n * region of text can differ from the full text value. Since cursor movement already handles\n * scrolling for that region change, then we only need to update the new text. Otherwise, we\n * also need to scroll and update the new text.\n */\n updateTextDisplay(fromCursorMove = false) {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description (async) Handles setting up this textarea once it is connected to run as an entity component.\n */\n async connected() {\n await super.connected();\n\n // Cursor Setup\n this.cursorWidth = 0.002;\n this.cursorHeight = mrjsUtils.css.pxToThree(mrjsUtils.css.extractNumFromPixelStr(this.compStyle.fontSize));\n this._createCursorObject();\n // initial style\n this.cursor.position.z += 0.001;\n this.cursor.visible = false;\n // We store this for the geometry so we can do our geometry vs web origin calculations\n // more easily as well. We update this based on the geometry's own changes.\n //\n // Set as 0,0,0 to start, and updated when the geometry updates in case it changes in 3d space.\n this.cursorStartingPosition = new three__WEBPACK_IMPORTED_MODULE_1__.Vector3(0, 0, 0);\n this.object3D.add(this.cursor);\n\n // DOM\n this.createHiddenInputElement();\n // this.fillInHiddenInputElementWithUserData(); // TODO - need good list of defaults\n\n // Make it trigger happy\n this.setupEventListeners();\n\n // Updates for baseline visual\n this.triggerGeometryStyleUpdate();\n this.triggerTextStyleUpdate();\n\n // All items should start out as 'not selected'\n // unless noted otherwise.\n if (!this.hiddenInput.getAttribute('autofocus') ?? false) {\n this._blur();\n }\n\n // Handle any placeholder setup s.t. it can be overwritten easily.\n if (this.hiddenInput.getAttribute('placeholder') ?? false) {\n this.textObj.text = this.hiddenInput.getAttribute('placeholder');\n }\n\n // TODO - this needs better boolean naming\n if (this.hasTextSubsetForVerticalScrolling) {\n this.verticalTextObjStartLineIndex = 0;\n this.verticalTextObjEndLineIndex = 0;\n }\n }\n\n /**\n * @function\n * @description Internal function used to setup the cursor object and associated variables\n * needed during runtime. Sets the cursor geometry based on dev updated cursorWidth and\n * cursorHeight MRTextInputEntity variables.\n */\n _createCursorObject() {\n if (!this.cursor) {\n // Setup basic cursor info and material for if it was reset.\n this.cursor = new three__WEBPACK_IMPORTED_MODULE_1__.Mesh();\n const material = new three__WEBPACK_IMPORTED_MODULE_1__.MeshBasicMaterial({\n color: 0x000000,\n side: three__WEBPACK_IMPORTED_MODULE_1__.DoubleSide,\n });\n this.cursor.material = material;\n }\n if (this.cursor.geometry !== undefined) {\n // Handle geometry reclearing\n this.cursor.geometry.dispose();\n }\n // Setup basic cursor geometry\n this.cursor.geometry = new three__WEBPACK_IMPORTED_MODULE_1__.PlaneGeometry(this.cursorWidth, this.cursorHeight);\n this.cursor.geometry.needsUpdate = true;\n }\n\n /**\n * @function\n * @description Internal function used to setup the cursor object and associated variables\n * needed during runtime. User can pass in a new height directly or the function checks\n * whether cursor height should be updated based on fontSize compared to line height\n * and other aspects.\n * @param {number} newHeight - an optional parameter to be used as the cursor's new height.\n */\n _updateCursorSize(newHeight) {\n const cursorVisibleHeight = newHeight ?? this.textObj.fontSize * this.lineHeight;\n if (this.cursor.geometry.parameters.height != cursorVisibleHeight) {\n this.cursorHeight = cursorVisibleHeight;\n this._createCursorObject();\n }\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Called by the keydown event trigger.\n * @param {event} event - the keydown event\n */\n handleKeydown(event) {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description Called by the mouse click event trigger. Handles determining the\n * caret position based on the 3D textObj to hiddenInput DOM position conversion.\n * @param {event} event - the mouseclick event\n */\n handleMouseClick(event) {\n console.log(event);\n // Convert isx position from world position to local:\n // - make sure textObj has updated matrices so we're not calculating info wrong\n // - note: textObj doesnt need sync\n this.textObj.updateMatrixWorld(true);\n const inverseMatrixWorld = new three__WEBPACK_IMPORTED_MODULE_1__.Matrix4().copy(this.textObj.matrixWorld).invert();\n const localPosition = inverseMatrixWorld * event.worldPosition;\n\n // update cursor position based on click\n // TODO - hitting an issue where carret's hit locations are undefined (ie not hit when it should)\n // so the return is defaulting to 0 index. Tried also the below of maybe with textObj but then it\n // never actually runs the code either (bc no text sync update is needed).\n // this.textObj.sync(() => {\n // const caret = getCaretAtPoint(this.textObj.textRenderInfo, localPosition.x, localPosition.y);\n // console.log('caret position: ', caret);\n // this.hiddenInput.selectionStart = caret.charIndex;\n // this.updateCursorPosition();\n // });\n const caret = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getCaretAtPoint)(this.textObj.textRenderInfo, localPosition.x, localPosition.y);\n console.log('caret position: ', caret);\n this.hiddenInput.selectionStart = caret.charIndex;\n this.updateCursorPosition();\n }\n\n /**\n * @function\n * @description Called by the focus event trigger and in other 'focus' situations. We use the\n * private version of this function signature to not hit the intersection of the actual 'focus()'\n * event naming that we have connected. See 'setupEventListeners()' description for more info.\n * @param {boolean} isPureFocusEvent - Boolean to allow us to update the cursor position with this function\n * directly. Otherwise, we assume there's other things happening after focus was called as part of the event\n * and that the cursor position will be handled there instead.\n */\n _focus(isPureFocusEvent = false) {\n if (!this.hiddenInput) {\n return;\n }\n this.hiddenInput.focus();\n\n if (isPureFocusEvent) {\n // Only want to update cursor and selection position if\n // this is a pure focus event; otherwise, we're assuming\n // the other event will position those properly (so that\n // we dont do redundant positioning here and then there as well).\n this.hiddenInput.selectionStart = this.hiddenInput.value.length;\n this.updateCursorPosition();\n }\n\n this.cursor.visible = true;\n }\n\n /**\n * @function\n * @description Called by the blur event trigger and in other 'blur' situations. We use the\n * private version of this function signature to not hit the intersection of the actual 'blur()'\n * event naming that we have connected. See 'setupEventListeners()' description for more info.\n */\n _blur() {\n if (!this.hiddenInput) {\n return;\n }\n this.hiddenInput.blur();\n\n this.cursor.visible = false;\n }\n\n // TODO - better name?\n /**\n * @function\n * @description Getter for whether this textinput should handle vertical scrolling or not.\n * @returns {boolean} true if it should be handled, false otherwise\n */\n get hasTextSubsetForVerticalScrolling() {\n // Leaving this as a function instead of a pure boolean if in case it's dependant\n // on certain parameters changing that need to be recalculated depending on the input\n // setup, etc.\n mrjsUtils.error.emptyParentFunction();\n return false;\n }\n\n // TODO - better name?\n /**\n * @function\n * @description Getter for whether this textinput should handle horizontal scrolling or not.\n * @returns {boolean} true if it should be handled, false otherwise\n */\n get hasTextSubsetForHorizontalScrolling() {\n // Leaving this as a function instead of a pure boolean if in case it's dependant\n // on certain parameters changing that need to be recalculated depending on the input\n // setup, etc.\n mrjsUtils.error.emptyParentFunction();\n return false;\n }\n\n /**\n * @function\n * @description Getter for a commonly needed attribute: 'disabled' for whether this input is still being updated.\n * @returns {boolean} true if disabled, false otherwise\n */\n get inputIsDisabled() {\n return this.hiddenInput.getAttribute('disabled') ?? false;\n }\n\n /**\n * @function\n * @description Getter for a commonly needed attribute: 'readonly' for whether this input's text can still be changed.\n * @returns {boolean} true if readonly, false otherwise\n */\n get inputIsReadOnly() {\n return this.hiddenInput.getAttribute('readonly') ?? false;\n }\n\n /**\n * @function\n * @description Connecting the event listeners to the actual functions that handle them. Includes\n * additional calls where necessary.\n *\n * Since we want the text input children to be able\n * to override the parent function event triggers,\n * separating them into an actual function here\n * and calling them manually instead of doing the pure\n * 'functionname () => {} event type setup'. This manual\n * connection allows us to call super.func() for event\n * functions; otherwise, theyre not accessible nor implemented\n * in the subclasses.\n */\n setupEventListeners() {\n // Blur events\n this.addEventListener('blur', () => {\n this._blur();\n });\n\n // Pure Focus Events\n this.addEventListener('focus', () => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n this._focus(true);\n });\n this.addEventListener('click', () => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n this._focus(true);\n });\n\n // Focus and Handle Event\n this.addEventListener('touchstart', (event) => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n this._focus();\n this.handleMouseClick(event);\n });\n\n // Keyboard events to capture text in the hidden input.\n this.hiddenInput.addEventListener('input', (event) => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n\n // Input captures all main text character inputs\n // BUT it does not capture arrow keys, so we handle\n // those specifically by the 'keydown' event.\n //\n // We handle all the rest by relying on the internal\n // 'hiddenInput's update system so we dont have to\n // manage as many things directly ourselves.\n\n this.updateTextDisplay(false);\n this.updateCursorPosition(false);\n });\n this.hiddenInput.addEventListener('keydown', (event) => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n\n // Only using keydown for arrow keys, everything else is\n // handled by the input event - check the comment there\n // for more reasoning.\n\n if (event.key == 'ArrowUp' || event.key == 'ArrowDown' || event.key == 'ArrowLeft' || event.key == 'ArrowRight') {\n this.handleKeydown(event);\n }\n });\n\n // Separate trigger call just in case.\n this.addEventListener('update-cursor-position', () => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n\n this.updateCursorPosition();\n });\n }\n\n // TODO - see note where this is called - need to rethink small part of indexing\n // logic so this manual aspect is not necessary.\n /**\n * @function\n * @description Helper function for `handleKeyDown` and `updateCursorPosition` when\n * handling textObj.\n * @param {number} lineIndex - the ending line index non-inclusive of the summation.\n * @param {Array} allLines - the array of line strings\n * @returns {number} length of summed string.\n */\n _totalLengthUpToLineIndex(lineIndex, allLines) {\n let totalLengthTolineIndex = 0;\n for (let i = 0; i < lineIndex; ++i) {\n totalLengthTolineIndex += allLines[i].length + 1; // one additional for '\\n' char\n }\n // TODO - will we need a check/fix to handle the case where the last index\n // may or may not have a '\\n' in it?\n return totalLengthTolineIndex;\n }\n\n // TODO - see note where this is called - need to rethink small part of indexing\n // logic so this manual aspect is not necessary.\n /**\n * @function\n * @description Helper function for `handleKeyDown` and `updateCursorPosition` when\n * handling textObj.\n * @param {number} lineIndexStart - the starting line index inclusive of the summation.\n * @param {number} lineIndexEnd - the starting line index non-inclusive of the summation.\n * @param {Array} allLines - the array of line strings\n * @returns {number} length of summed string.\n */\n _totalLengthBetweenLineIndices(lineIndexStart, lineIndexEnd, allLines) {\n let totalLengthTolineIndex = 0;\n for (let i = lineIndexStart; i < lineIndexEnd; ++i) {\n totalLengthTolineIndex += allLines[i].length + 1; // one additional for '\\n' char\n }\n // TODO - will we need a check/fix to handle the case where the last index\n // may or may not have a '\\n' in it?\n return totalLengthTolineIndex;\n }\n\n /**\n * @function\n * @description Updates the cursor position based on click and selection location.\n * @param {boolean} fromCursorMove - false by default. Used to determine if we need to run\n * based off a text object update sync or we can directly grab information. This requirement\n * occurs because the sync isnt usable if no text content changed.\n *\n * Note: this function does not change anything about the this.hiddenInput.selectionStart nor\n * this.hiddenInput.selectionEnd. Those values should be changed prior to this function being\n * called.\n */\n updateCursorPosition(fromCursorMove = false) {\n // TODO - QUESTION: handle '\\n' --> as '/\\r?\\n/' for crossplatform compat\n // does the browser handle this for us?\n\n const updateBasedOnSelectionRects = (cursorIndex) => {\n // Setup variables for calculations.\n let textBeforeCursor = this.hiddenInput.value.substring(0, cursorIndex);\n let allLines = this.hiddenInput.value.split('\\n');\n let linesBeforeCursor = textBeforeCursor.split('\\n');\n let cursorIsOnLineIndex = linesBeforeCursor.length - 1;\n\n let rectX = undefined;\n let rectY = undefined;\n let rect = undefined;\n\n // Setup textObj variables for calculations\n let cursorIsOnTextObjLineIndex = cursorIsOnLineIndex - this.verticalTextObjStartLineIndex;\n // TODO - there needs to be a cleaner way to do this than summing up every time\n let lengthToCursorTextObjStartLineIndex = this._totalLengthUpToLineIndex(this.verticalTextObjStartLineIndex, allLines);\n let lengthToTextObjCursorLine = this._totalLengthBetweenLineIndices(this.verticalTextObjStartLineIndex, cursorIsOnLineIndex, allLines);\n // note: add one to start it on the actual start line so we can index cursor Index at 0 if beg of line\n let cursorIndexWithinTextObj = cursorIndex - (lengthToCursorTextObjStartLineIndex + 1);\n\n const prevIsNewlineChar = '\\n' === textBeforeCursor.charAt(textBeforeCursor.length - 1);\n if (prevIsNewlineChar) {\n // When on newline char, hiddenInput puts selection at end of newline char,\n // not beginning of next line. Make sure cursor visual is at beginning\n // of the next line without moving selection point.\n //\n // \"\"\"\n // This is an example of text\\n\n // the way troika handles it\n // \"\"\"\n //\n // In the below, using (*) to denote the 'you are here'.\n const isFakeNewLine = cursorIsOnLineIndex == allLines.length - 1;\n let indexOfBegOfLine = lengthToTextObjCursorLine;\n if (isFakeNewLine) {\n // \"\"\"\n // This is an example of text\\n\n // the way troika handles it\\n(*)\n // \"\"\"\n //\n // Special case where next line doesnt exist yet, fake it sitting below with our\n // current line's information.\n let usingIndex = indexOfBegOfLine - 1;\n let selectionRects = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getSelectionRects)(this.textObj.textRenderInfo, usingIndex, usingIndex + 1);\n // rect information for use in cursor positioning\n rect = selectionRects[0];\n rectY = rect.bottom - this.cursorHeight;\n } else {\n // \"\"\"\n // This is an example of text\\n(*)\n // the way troika handles it\n // \"\"\"\n let usingIndex = cursorIndexWithinTextObj;\n let selectionRects = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getSelectionRects)(this.textObj.textRenderInfo, usingIndex, usingIndex + 1);\n rect = selectionRects[0];\n rectY = rect.bottom;\n }\n rectX = 0;\n } else {\n // default\n let usingIndex = cursorIndexWithinTextObj;\n let selectionRects = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getSelectionRects)(this.textObj.textRenderInfo, usingIndex, usingIndex + 1);\n // rect information for use in cursor positioning\n rect = selectionRects[0];\n rectX = rect.right;\n rectY = rect.bottom;\n }\n\n this._updateCursorSize();\n\n // Add the cursor dimension info to the position s.t. it doesnt touch the text itself. We want\n // a little bit of buffer room.\n const cursorXOffsetPosition = rectX + this.cursorWidth;\n const cursorYOffsetPosition = rectY + this.cursorHeight;\n\n // Update the cursor's 3D position\n this.cursor.position.x = this.cursorStartingPosition.x + cursorXOffsetPosition;\n this.cursor.position.y = this.cursorStartingPosition.y + cursorYOffsetPosition;\n this.cursor.visible = true;\n };\n\n // Check if we have any DOM element to work with.\n if (!this.hiddenInput) {\n return;\n }\n\n // Since no text is selected, this and selectionEnd are just the cursor position.\n // XXX - when we actually allow for selection in future, some of the below will need to\n // be thought through again.\n const cursorIndex = this.hiddenInput.selectionStart;\n\n // early escape for empty text based on hiddenInput top line\n if (cursorIndex == 0) {\n this._updateCursorSize();\n this.cursor.position.x = this.cursorStartingPosition.x;\n this.cursor.position.y = this.cursorStartingPosition.y;\n this.cursor.visible = true;\n return;\n }\n\n // Separating textObj sync from the cursor update based on rects\n // since textObj sync resolves when there's actual changes to the\n // object. Otherwise, it'll hang and never hit the update function.\n if (fromCursorMove) {\n updateBasedOnSelectionRects(cursorIndex);\n } else {\n this.textObj.sync(() => {\n updateBasedOnSelectionRects(cursorIndex);\n });\n }\n }\n}\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRTextInputEntity.js?"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRTextInputEntity: () => (/* binding */ MRTextInputEntity)\n/* harmony export */ });\n/* harmony import */ var troika_three_text__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! troika-three-text */ \"./node_modules/troika-three-text/dist/troika-three-text.esm.js\");\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var mrjs_core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRTextEntity */ \"./src/core/entities/MRTextEntity.js\");\n\n\n\n\n\n\n/**\n * @class MRTextInputEntity\n * @classdesc Base text inpu entity represented in 3D space. `mr-text-input`\n * @augments MRTextEntity\n */\nclass MRTextInputEntity extends mrjs_core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_0__.MRTextEntity {\n /**\n * @class\n * @description Constructor for the MRTextInputEntity entity component.\n */\n constructor() {\n super();\n this.attachShadow({ mode: 'open' });\n }\n\n /**\n * @function\n * @description Gets the value of the text for the current hiddenInput DOM object\n * @returns {string} value - the text value of the current hiddenInput DOM object\n */\n get value() {\n return this.hiddenInput.value;\n }\n\n /**\n * @function\n * @description Sets the value of the text for the current hiddenInput DOM object\n */\n set value(val) {\n this.hiddenInput.value = val;\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Called by connected to make sure\n * the hiddenInput dom element is created as expected.\n */\n createHiddenInputElement() {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Called by connected after\n * createHiddenInputElement to fill it in with the user's given\n * attribute information.\n */\n fillInHiddenInputElementWithUserData() {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Used on event trigger to\n * update the textObj visual based on the hiddenInput DOM element.\n * @param {boolean} fromCursorMove - default set as false if not supplied. See `MRTextArea`\n * and `MRTextField` as examples. This param is helpful for cases where the visible\n * region of text can differ from the full text value. Since cursor movement already handles\n * scrolling for that region change, then we only need to update the new text. Otherwise, we\n * also need to scroll and update the new text.\n */\n updateTextDisplay(fromCursorMove = false) {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description (async) Handles setting up this textarea once it is connected to run as an entity component.\n */\n async connected() {\n // await super.connected(); // TODO - uncomment this in the textfield pr if this is actually needed, but atm it's not since not exposed.\n\n // Cursor Setup\n this.cursorWidth = 0.002;\n this.cursorHeight = mrjsUtils.css.pxToThree(mrjsUtils.css.extractNumFromPixelStr(this.compStyle.fontSize));\n this._createCursorObject();\n // initial style\n this.cursor.position.z += 0.001;\n this.cursor.visible = false;\n // We store this for the geometry so we can do our geometry vs web origin calculations\n // more easily as well. We update this based on the geometry's own changes.\n //\n // Set as 0,0,0 to start, and updated when the geometry updates in case it changes in 3d space.\n this.cursorStartingPosition = new three__WEBPACK_IMPORTED_MODULE_1__.Vector3(0, 0, 0);\n this.object3D.add(this.cursor);\n\n // DOM\n this.createHiddenInputElement();\n // this.fillInHiddenInputElementWithUserData(); // TODO - need good list of defaults\n\n // Make it trigger happy\n this.setupEventListeners();\n\n // Updates for baseline visual\n this.triggerGeometryStyleUpdate();\n this.triggerTextStyleUpdate();\n\n // All items should start out as 'not selected'\n // unless noted otherwise.\n if (!this.hiddenInput.getAttribute('autofocus') ?? false) {\n this._blur();\n }\n\n // Handle any placeholder setup s.t. it can be overwritten easily.\n if (this.hiddenInput.getAttribute('placeholder') ?? false) {\n this.textObj.text = this.hiddenInput.getAttribute('placeholder');\n }\n\n // TODO - this needs better boolean naming\n if (this.hasTextSubsetForVerticalScrolling) {\n this.verticalTextObjStartLineIndex = 0;\n this.verticalTextObjEndLineIndex = 0;\n }\n }\n\n /**\n * @function\n * @description Internal function used to setup the cursor object and associated variables\n * needed during runtime. Sets the cursor geometry based on dev updated cursorWidth and\n * cursorHeight MRTextInputEntity variables.\n */\n _createCursorObject() {\n if (!this.cursor) {\n // Setup basic cursor info and material for if it was reset.\n this.cursor = new three__WEBPACK_IMPORTED_MODULE_1__.Mesh();\n const material = new three__WEBPACK_IMPORTED_MODULE_1__.MeshBasicMaterial({\n color: 0x000000,\n side: three__WEBPACK_IMPORTED_MODULE_1__.DoubleSide,\n });\n this.cursor.material = material;\n }\n if (this.cursor.geometry !== undefined) {\n // Handle geometry reclearing\n this.cursor.geometry.dispose();\n }\n // Setup basic cursor geometry\n this.cursor.geometry = new three__WEBPACK_IMPORTED_MODULE_1__.PlaneGeometry(this.cursorWidth, this.cursorHeight);\n this.cursor.geometry.needsUpdate = true;\n }\n\n /**\n * @function\n * @description Internal function used to setup the cursor object and associated variables\n * needed during runtime. User can pass in a new height directly or the function checks\n * whether cursor height should be updated based on fontSize compared to line height\n * and other aspects.\n * @param {number} newHeight - an optional parameter to be used as the cursor's new height.\n */\n _updateCursorSize(newHeight) {\n const cursorVisibleHeight = newHeight ?? this.textObj.fontSize * this.lineHeight;\n if (this.cursor.geometry.parameters.height != cursorVisibleHeight) {\n this.cursorHeight = cursorVisibleHeight;\n this._createCursorObject();\n }\n }\n\n /**\n * @function\n * @description Function to be overwritten by children. Called by the keydown event trigger.\n * @param {event} event - the keydown event\n */\n handleKeydown(event) {\n mrjsUtils.error.emptyParentFunction();\n }\n\n /**\n * @function\n * @description Called by the mouse click event trigger. Handles determining the\n * caret position based on the 3D textObj to hiddenInput DOM position conversion.\n * @param {event} event - the mouseclick event\n */\n handleMouseClick(event) {\n console.log(event);\n // Convert isx position from world position to local:\n // - make sure textObj has updated matrices so we're not calculating info wrong\n // - note: textObj doesnt need sync\n this.textObj.updateMatrixWorld(true);\n const inverseMatrixWorld = new three__WEBPACK_IMPORTED_MODULE_1__.Matrix4().copy(this.textObj.matrixWorld).invert();\n const localPosition = inverseMatrixWorld * event.worldPosition;\n\n // update cursor position based on click\n // TODO - hitting an issue where carret's hit locations are undefined (ie not hit when it should)\n // so the return is defaulting to 0 index. Tried also the below of maybe with textObj but then it\n // never actually runs the code either (bc no text sync update is needed).\n // this.textObj.sync(() => {\n // const caret = getCaretAtPoint(this.textObj.textRenderInfo, localPosition.x, localPosition.y);\n // console.log('caret position: ', caret);\n // this.hiddenInput.selectionStart = caret.charIndex;\n // this.updateCursorPosition();\n // });\n const caret = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getCaretAtPoint)(this.textObj.textRenderInfo, localPosition.x, localPosition.y);\n console.log('caret position: ', caret);\n this.hiddenInput.selectionStart = caret.charIndex;\n this.updateCursorPosition();\n }\n\n /**\n * @function\n * @description Called by the focus event trigger and in other 'focus' situations. We use the\n * private version of this function signature to not hit the intersection of the actual 'focus()'\n * event naming that we have connected. See 'setupEventListeners()' description for more info.\n * @param {boolean} isPureFocusEvent - Boolean to allow us to update the cursor position with this function\n * directly. Otherwise, we assume there's other things happening after focus was called as part of the event\n * and that the cursor position will be handled there instead.\n */\n _focus(isPureFocusEvent = false) {\n if (!this.hiddenInput) {\n return;\n }\n this.hiddenInput.focus();\n\n if (isPureFocusEvent) {\n // Only want to update cursor and selection position if\n // this is a pure focus event; otherwise, we're assuming\n // the other event will position those properly (so that\n // we dont do redundant positioning here and then there as well).\n this.hiddenInput.selectionStart = this.hiddenInput.value.length;\n this.updateCursorPosition();\n }\n\n this.cursor.visible = true;\n }\n\n /**\n * @function\n * @description Called by the blur event trigger and in other 'blur' situations. We use the\n * private version of this function signature to not hit the intersection of the actual 'blur()'\n * event naming that we have connected. See 'setupEventListeners()' description for more info.\n */\n _blur() {\n if (!this.hiddenInput) {\n return;\n }\n this.hiddenInput.blur();\n\n this.cursor.visible = false;\n }\n\n // TODO - better name?\n /**\n * @function\n * @description Getter for whether this textinput should handle vertical scrolling or not.\n * @returns {boolean} true if it should be handled, false otherwise\n */\n get hasTextSubsetForVerticalScrolling() {\n // Leaving this as a function instead of a pure boolean if in case it's dependant\n // on certain parameters changing that need to be recalculated depending on the input\n // setup, etc.\n mrjsUtils.error.emptyParentFunction();\n return false;\n }\n\n // TODO - better name?\n /**\n * @function\n * @description Getter for whether this textinput should handle horizontal scrolling or not.\n * @returns {boolean} true if it should be handled, false otherwise\n */\n get hasTextSubsetForHorizontalScrolling() {\n // Leaving this as a function instead of a pure boolean if in case it's dependant\n // on certain parameters changing that need to be recalculated depending on the input\n // setup, etc.\n mrjsUtils.error.emptyParentFunction();\n return false;\n }\n\n /**\n * @function\n * @description Getter for a commonly needed attribute: 'disabled' for whether this input is still being updated.\n * @returns {boolean} true if disabled, false otherwise\n */\n get inputIsDisabled() {\n return this.hiddenInput.getAttribute('disabled') ?? false;\n }\n\n /**\n * @function\n * @description Getter for a commonly needed attribute: 'readonly' for whether this input's text can still be changed.\n * @returns {boolean} true if readonly, false otherwise\n */\n get inputIsReadOnly() {\n return this.hiddenInput.getAttribute('readonly') ?? false;\n }\n\n /**\n * @function\n * @description Connecting the event listeners to the actual functions that handle them. Includes\n * additional calls where necessary.\n *\n * Since we want the text input children to be able\n * to override the parent function event triggers,\n * separating them into an actual function here\n * and calling them manually instead of doing the pure\n * 'functionname () => {} event type setup'. This manual\n * connection allows us to call super.func() for event\n * functions; otherwise, theyre not accessible nor implemented\n * in the subclasses.\n */\n setupEventListeners() {\n // Blur events\n this.addEventListener('blur', () => {\n this._blur();\n });\n\n // Pure Focus Events\n this.addEventListener('focus', () => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n this._focus(true);\n });\n this.addEventListener('click', () => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n this._focus(true);\n });\n\n // Focus and Handle Event\n this.addEventListener('touchstart', (event) => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n this._focus();\n this.handleMouseClick(event);\n });\n\n // Keyboard events to capture text in the hidden input.\n this.hiddenInput.addEventListener('input', (event) => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n\n // Input captures all main text character inputs\n // BUT it does not capture arrow keys, so we handle\n // those specifically by the 'keydown' event.\n //\n // We handle all the rest by relying on the internal\n // 'hiddenInput's update system so we dont have to\n // manage as many things directly ourselves.\n\n this.updateTextDisplay(false);\n this.updateCursorPosition(false);\n });\n this.hiddenInput.addEventListener('keydown', (event) => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n\n // Only using keydown for arrow keys, everything else is\n // handled by the input event - check the comment there\n // for more reasoning.\n\n if (event.key == 'ArrowUp' || event.key == 'ArrowDown' || event.key == 'ArrowLeft' || event.key == 'ArrowRight') {\n this.handleKeydown(event);\n }\n });\n\n // Separate trigger call just in case.\n this.addEventListener('update-cursor-position', () => {\n if (this.inputIsDisabled || this.inputIsReadOnly) {\n return;\n }\n\n this.updateCursorPosition();\n });\n }\n\n // TODO - see note where this is called - need to rethink small part of indexing\n // logic so this manual aspect is not necessary.\n /**\n * @function\n * @description Helper function for `handleKeyDown` and `updateCursorPosition` when\n * handling textObj.\n * @param {number} lineIndex - the ending line index non-inclusive of the summation.\n * @param {Array} allLines - the array of line strings\n * @returns {number} length of summed string.\n */\n _totalLengthUpToLineIndex(lineIndex, allLines) {\n let totalLengthTolineIndex = 0;\n for (let i = 0; i < lineIndex; ++i) {\n totalLengthTolineIndex += allLines[i].length + 1; // one additional for '\\n' char\n }\n // TODO - will we need a check/fix to handle the case where the last index\n // may or may not have a '\\n' in it?\n return totalLengthTolineIndex;\n }\n\n // TODO - see note where this is called - need to rethink small part of indexing\n // logic so this manual aspect is not necessary.\n /**\n * @function\n * @description Helper function for `handleKeyDown` and `updateCursorPosition` when\n * handling textObj.\n * @param {number} lineIndexStart - the starting line index inclusive of the summation.\n * @param {number} lineIndexEnd - the starting line index non-inclusive of the summation.\n * @param {Array} allLines - the array of line strings\n * @returns {number} length of summed string.\n */\n _totalLengthBetweenLineIndices(lineIndexStart, lineIndexEnd, allLines) {\n let totalLengthTolineIndex = 0;\n for (let i = lineIndexStart; i < lineIndexEnd; ++i) {\n totalLengthTolineIndex += allLines[i].length + 1; // one additional for '\\n' char\n }\n // TODO - will we need a check/fix to handle the case where the last index\n // may or may not have a '\\n' in it?\n return totalLengthTolineIndex;\n }\n\n /**\n * @function\n * @description Updates the cursor position based on click and selection location.\n * @param {boolean} fromCursorMove - false by default. Used to determine if we need to run\n * based off a text object update sync or we can directly grab information. This requirement\n * occurs because the sync isnt usable if no text content changed.\n *\n * Note: this function does not change anything about the this.hiddenInput.selectionStart nor\n * this.hiddenInput.selectionEnd. Those values should be changed prior to this function being\n * called.\n */\n updateCursorPosition(fromCursorMove = false) {\n // TODO - QUESTION: handle '\\n' --> as '/\\r?\\n/' for crossplatform compat\n // does the browser handle this for us?\n\n const updateBasedOnSelectionRects = (cursorIndex) => {\n // Setup variables for calculations.\n let textBeforeCursor = this.hiddenInput.value.substring(0, cursorIndex);\n let allLines = this.hiddenInput.value.split('\\n');\n let linesBeforeCursor = textBeforeCursor.split('\\n');\n let cursorIsOnLineIndex = linesBeforeCursor.length - 1;\n\n let rectX = undefined;\n let rectY = undefined;\n let rect = undefined;\n\n // Setup textObj variables for calculations\n let cursorIsOnTextObjLineIndex = cursorIsOnLineIndex - this.verticalTextObjStartLineIndex;\n // TODO - there needs to be a cleaner way to do this than summing up every time\n let lengthToCursorTextObjStartLineIndex = this._totalLengthUpToLineIndex(this.verticalTextObjStartLineIndex, allLines);\n let lengthToTextObjCursorLine = this._totalLengthBetweenLineIndices(this.verticalTextObjStartLineIndex, cursorIsOnLineIndex, allLines);\n // note: add one to start it on the actual start line so we can index cursor Index at 0 if beg of line\n let cursorIndexWithinTextObj = cursorIndex - (lengthToCursorTextObjStartLineIndex + 1);\n\n const prevIsNewlineChar = '\\n' === textBeforeCursor.charAt(textBeforeCursor.length - 1);\n if (prevIsNewlineChar) {\n // When on newline char, hiddenInput puts selection at end of newline char,\n // not beginning of next line. Make sure cursor visual is at beginning\n // of the next line without moving selection point.\n //\n // \"\"\"\n // This is an example of text\\n\n // the way troika handles it\n // \"\"\"\n //\n // In the below, using (*) to denote the 'you are here'.\n const isFakeNewLine = cursorIsOnLineIndex == allLines.length - 1;\n let indexOfBegOfLine = lengthToTextObjCursorLine;\n if (isFakeNewLine) {\n // \"\"\"\n // This is an example of text\\n\n // the way troika handles it\\n(*)\n // \"\"\"\n //\n // Special case where next line doesnt exist yet, fake it sitting below with our\n // current line's information.\n let usingIndex = indexOfBegOfLine - 1;\n let selectionRects = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getSelectionRects)(this.textObj.textRenderInfo, usingIndex, usingIndex + 1);\n // rect information for use in cursor positioning\n rect = selectionRects[0];\n rectY = rect.bottom - this.cursorHeight;\n } else {\n // \"\"\"\n // This is an example of text\\n(*)\n // the way troika handles it\n // \"\"\"\n let usingIndex = cursorIndexWithinTextObj;\n let selectionRects = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getSelectionRects)(this.textObj.textRenderInfo, usingIndex, usingIndex + 1);\n rect = selectionRects[0];\n rectY = rect.bottom;\n }\n rectX = 0;\n } else {\n // default\n let usingIndex = cursorIndexWithinTextObj;\n let selectionRects = (0,troika_three_text__WEBPACK_IMPORTED_MODULE_2__.getSelectionRects)(this.textObj.textRenderInfo, usingIndex, usingIndex + 1);\n // rect information for use in cursor positioning\n rect = selectionRects[0];\n rectX = rect.right;\n rectY = rect.bottom;\n }\n\n this._updateCursorSize();\n\n // Add the cursor dimension info to the position s.t. it doesnt touch the text itself. We want\n // a little bit of buffer room.\n const cursorXOffsetPosition = rectX + this.cursorWidth;\n const cursorYOffsetPosition = rectY + this.cursorHeight;\n\n // Update the cursor's 3D position\n this.cursor.position.x = this.cursorStartingPosition.x + cursorXOffsetPosition;\n this.cursor.position.y = this.cursorStartingPosition.y + cursorYOffsetPosition;\n this.cursor.visible = true;\n };\n\n // Check if we have any DOM element to work with.\n if (!this.hiddenInput) {\n return;\n }\n\n // Since no text is selected, this and selectionEnd are just the cursor position.\n // XXX - when we actually allow for selection in future, some of the below will need to\n // be thought through again.\n const cursorIndex = this.hiddenInput.selectionStart;\n\n // early escape for empty text based on hiddenInput top line\n if (cursorIndex == 0) {\n this._updateCursorSize();\n this.cursor.position.x = this.cursorStartingPosition.x;\n this.cursor.position.y = this.cursorStartingPosition.y;\n this.cursor.visible = true;\n return;\n }\n\n // Separating textObj sync from the cursor update based on rects\n // since textObj sync resolves when there's actual changes to the\n // object. Otherwise, it'll hang and never hit the update function.\n if (fromCursorMove) {\n updateBasedOnSelectionRects(cursorIndex);\n } else {\n this.textObj.sync(() => {\n updateBasedOnSelectionRects(cursorIndex);\n });\n }\n }\n}\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRTextInputEntity.js?"); /***/ }), @@ -817,7 +817,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpac /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRVideoEntity: () => (/* binding */ MRVideoEntity)\n/* harmony export */ });\n/* harmony import */ var mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRMediaEntity */ \"./src/core/entities/MRMediaEntity.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n\n\n\n\n\n\n/**\n * @class MRVideoEntity\n * @classdesc Base html video represented in 3D space. `mr-video`\n * @augments MRMediaEntity\n */\nclass MRVideoEntity extends mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__.MRMediaEntity {\n /**\n * @class\n * @description Constructs a base video entity using a UIPlane and other 3D elements as necessary.\n */\n constructor() {\n super();\n this.object3D.name = 'video';\n this.playing = false;\n }\n\n /**\n * @function\n * @description Calculates the width of the video based on the video tag itself\n * @returns {number} - the resolved width\n */\n get mediaWidth() {\n return this.media.videoWidth;\n }\n\n /**\n * @function\n * @description Calculates the height of the video based on the video tag itself\n * @returns {number} - the resolved height\n */\n get mediaHeight() {\n return this.media.videoHeight;\n }\n\n /**\n * @function\n * @description Loads the associated video into 3D based on its html properties.\n */\n loadMediaTexture() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.material\n .loadVideoTextureAsync(this.media)\n .then((texture) => {\n this.texture = texture;\n this.object3D.material.map = texture;\n this.playing = true; // since we have videos auto play on silent to start\n })\n .catch((error) => {\n console.error('Error loading texture:', error);\n });\n }\n\n /**\n * @function\n * @description (async) handles setting up this video and associated 3D geometry style (from css) once it is connected to run as an entity component.\n */\n async connected() {\n await super.connected();\n this.media = document.createElement('video');\n this.media.setAttribute('crossorigin', 'anonymous');\n\n super.connected();\n }\n\n /**\n * @function\n * @description Sets the srcObject of the video media (since it uses 'srcObject' instead of 'src' like other items).\n * @param {string} src - the string to the new source object we want\n */\n set srcObject(src) {\n this.media.srcObject = src;\n // on loadeddata event, update the objectFitDimensions\n this.media.addEventListener('loadeddata', () => {\n this.computeObjectFitDimensions();\n });\n }\n\n /**\n * @function\n * @description Plays the video in the shadow root\n */\n play() {\n this.media.play();\n this.playing = true;\n }\n\n /**\n * @function\n * @description Pauses the video in the shadow root\n */\n pause() {\n this.media.pause();\n this.playing = false;\n }\n}\n\ncustomElements.get('mr-video') || customElements.define('mr-video', MRVideoEntity);\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRVideoEntity.js?"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRVideoEntity: () => (/* binding */ MRVideoEntity)\n/* harmony export */ });\n/* harmony import */ var mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! mrjs/core/entities/MRMediaEntity */ \"./src/core/entities/MRMediaEntity.js\");\n/* harmony import */ var mrjs__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! mrjs */ \"./src/index.js\");\n\n\n\n\n\n\n/**\n * @class MRVideoEntity\n * @classdesc Base html video represented in 3D space. `mr-video`\n * @augments MRMediaEntity\n */\nclass MRVideoEntity extends mrjs_core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_0__.MRMediaEntity {\n /**\n * @class\n * @description Constructs a base video entity using a UIPlane and other 3D elements as necessary.\n */\n constructor() {\n super();\n this.object3D.name = 'video';\n this.playing = false;\n }\n\n /**\n * @function\n * @description Calculates the width of the video based on the video tag itself\n * @returns {number} - the resolved width\n */\n get mediaWidth() {\n return this.media.videoWidth;\n }\n\n /**\n * @function\n * @description Calculates the height of the video based on the video tag itself\n * @returns {number} - the resolved height\n */\n get mediaHeight() {\n return this.media.videoHeight;\n }\n\n /**\n * @function\n * @description Loads the associated video into 3D based on its html properties.\n */\n loadMediaTexture() {\n mrjs__WEBPACK_IMPORTED_MODULE_1__.mrjsUtils.material\n .loadVideoTextureAsync(this.media)\n .then((texture) => {\n this.texture = texture;\n this.object3D.material.map = texture;\n this.playing = true; // since we have videos auto play on silent to start\n })\n .catch((error) => {\n console.error('Error loading texture:', error);\n });\n }\n\n /**\n * @function\n * @description (async) handles setting up this video and associated 3D geometry style (from css) once it is connected to run as an entity component.\n */\n async connected() {\n this.media = document.createElement('video');\n this.media.setAttribute('crossorigin', 'anonymous');\n\n await super.connected();\n }\n\n /**\n * @function\n * @description Sets the srcObject of the video media (since it uses 'srcObject' instead of 'src' like other items).\n * @param {string} src - the string to the new source object we want\n */\n set srcObject(src) {\n this.media.srcObject = src;\n // on loadeddata event, update the objectFitDimensions\n this.media.addEventListener('loadeddata', () => {\n this.computeObjectFitDimensions();\n });\n }\n\n /**\n * @function\n * @description Plays the video in the shadow root\n */\n play() {\n this.media.play();\n this.playing = true;\n }\n\n /**\n * @function\n * @description Pauses the video in the shadow root\n */\n pause() {\n this.media.pause();\n this.playing = false;\n }\n}\n\ncustomElements.get('mr-video') || customElements.define('mr-video', MRVideoEntity);\n\n\n//# sourceURL=webpack://mrjs/./src/core/entities/MRVideoEntity.js?"); /***/ }), @@ -937,7 +937,7 @@ eval("/**\n * @global\n * @type {number}\n * @description the noted viewport hei /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRButtonEntity: () => (/* reexport safe */ _core_entities_MRButtonEntity__WEBPACK_IMPORTED_MODULE_12__.MRButtonEntity),\n/* harmony export */ MRDivEntity: () => (/* reexport safe */ _core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_13__.MRDivEntity),\n/* harmony export */ MREntity: () => (/* reexport safe */ _core_MREntity__WEBPACK_IMPORTED_MODULE_8__.MREntity),\n/* harmony export */ MRImageEntity: () => (/* reexport safe */ _core_entities_MRImageEntity__WEBPACK_IMPORTED_MODULE_15__.MRImageEntity),\n/* harmony export */ MRLightEntity: () => (/* reexport safe */ _core_entities_MRLightEntity__WEBPACK_IMPORTED_MODULE_16__.MRLightEntity),\n/* harmony export */ MRMediaEntity: () => (/* reexport safe */ _core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_17__.MRMediaEntity),\n/* harmony export */ MRModelEntity: () => (/* reexport safe */ _core_entities_MRModelEntity__WEBPACK_IMPORTED_MODULE_18__.MRModelEntity),\n/* harmony export */ MRPanelEntity: () => (/* reexport safe */ _core_entities_MRPanelEntity__WEBPACK_IMPORTED_MODULE_19__.MRPanelEntity),\n/* harmony export */ MRSkyBoxEntity: () => (/* reexport safe */ _core_entities_MRSkyBoxEntity__WEBPACK_IMPORTED_MODULE_20__.MRSkyBoxEntity),\n/* harmony export */ MRSystem: () => (/* reexport safe */ _core_MRSystem__WEBPACK_IMPORTED_MODULE_9__.MRSystem),\n/* harmony export */ MRTextAreaEntity: () => (/* reexport safe */ _core_entities_MRTextAreaEntity__WEBPACK_IMPORTED_MODULE_23__.MRTextAreaEntity),\n/* harmony export */ MRTextEntity: () => (/* reexport safe */ _core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_22__.MRTextEntity),\n/* harmony export */ MRTextFieldEntity: () => (/* reexport safe */ _core_entities_MRTextFieldEntity__WEBPACK_IMPORTED_MODULE_24__.MRTextFieldEntity),\n/* harmony export */ MRVideoEntity: () => (/* reexport safe */ _core_entities_MRVideoEntity__WEBPACK_IMPORTED_MODULE_26__.MRVideoEntity),\n/* harmony export */ MRVolumeEntity: () => (/* reexport safe */ _core_entities_MRVolumeEntity__WEBPACK_IMPORTED_MODULE_27__.MRVolumeEntity),\n/* harmony export */ Refractor: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.Refractor),\n/* harmony export */ THREE: () => (/* reexport module object */ three__WEBPACK_IMPORTED_MODULE_44__),\n/* harmony export */ Water: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.Water),\n/* harmony export */ WaterRefractionShader: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.WaterRefractionShader),\n/* harmony export */ WaterSystem: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.WaterSystem),\n/* harmony export */ mrjsUtils: () => (/* reexport safe */ _utils_index_js__WEBPACK_IMPORTED_MODULE_42__.mrjsUtils)\n/* harmony export */ });\n/* harmony import */ var _package_json__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../package.json */ \"./package.json\");\n/* harmony import */ var _package_json__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_package_json__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _defaultStyle_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./defaultStyle.css */ \"./src/defaultStyle.css\");\n/* harmony import */ var _global__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./global */ \"./src/global.js\");\n/* harmony import */ var _global__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(_global__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var _dataTypes_MRClippingGeometry__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./dataTypes/MRClippingGeometry */ \"./src/dataTypes/MRClippingGeometry.js\");\n/* harmony import */ var _dataTypes_MRPlane__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./dataTypes/MRPlane */ \"./src/dataTypes/MRPlane.js\");\n/* harmony import */ var _dataManagers_MRPlaneManager__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./dataManagers/MRPlaneManager */ \"./src/dataManagers/MRPlaneManager.js\");\n/* harmony import */ var _core_MRApp__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./core/MRApp */ \"./src/core/MRApp.js\");\n/* harmony import */ var _core_MRElement__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./core/MRElement */ \"./src/core/MRElement.js\");\n/* harmony import */ var _core_MREntity__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! mrjs/core/MREntity */ \"./src/core/MREntity.js\");\n/* harmony import */ var _core_MRSystem__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! mrjs/core/MRSystem */ \"./src/core/MRSystem.js\");\n/* harmony import */ var _core_user_MRHand__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ./core/user/MRHand */ \"./src/core/user/MRHand.js\");\n/* harmony import */ var _core_user_MRUser__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ./core/user/MRUser */ \"./src/core/user/MRUser.js\");\n/* harmony import */ var _core_entities_MRButtonEntity__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! mrjs/core/entities/MRButtonEntity */ \"./src/core/entities/MRButtonEntity.js\");\n/* harmony import */ var _core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! mrjs/core/entities/MRDivEntity */ \"./src/core/entities/MRDivEntity.js\");\n/* harmony import */ var _core_entities_MRHyperlinkEntity__WEBPACK_IMPORTED_MODULE_14__ = __webpack_require__(/*! mrjs/core/entities/MRHyperlinkEntity */ \"./src/core/entities/MRHyperlinkEntity.js\");\n/* harmony import */ var _core_entities_MRImageEntity__WEBPACK_IMPORTED_MODULE_15__ = __webpack_require__(/*! mrjs/core/entities/MRImageEntity */ \"./src/core/entities/MRImageEntity.js\");\n/* harmony import */ var _core_entities_MRLightEntity__WEBPACK_IMPORTED_MODULE_16__ = __webpack_require__(/*! mrjs/core/entities/MRLightEntity */ \"./src/core/entities/MRLightEntity.js\");\n/* harmony import */ var _core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_17__ = __webpack_require__(/*! mrjs/core/entities/MRMediaEntity */ \"./src/core/entities/MRMediaEntity.js\");\n/* harmony import */ var _core_entities_MRModelEntity__WEBPACK_IMPORTED_MODULE_18__ = __webpack_require__(/*! mrjs/core/entities/MRModelEntity */ \"./src/core/entities/MRModelEntity.js\");\n/* harmony import */ var _core_entities_MRPanelEntity__WEBPACK_IMPORTED_MODULE_19__ = __webpack_require__(/*! mrjs/core/entities/MRPanelEntity */ \"./src/core/entities/MRPanelEntity.js\");\n/* harmony import */ var _core_entities_MRSkyBoxEntity__WEBPACK_IMPORTED_MODULE_20__ = __webpack_require__(/*! mrjs/core/entities/MRSkyBoxEntity */ \"./src/core/entities/MRSkyBoxEntity.js\");\n/* harmony import */ var _core_entities_MRStatsEntity__WEBPACK_IMPORTED_MODULE_21__ = __webpack_require__(/*! ./core/entities/MRStatsEntity */ \"./src/core/entities/MRStatsEntity.js\");\n/* harmony import */ var _core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_22__ = __webpack_require__(/*! mrjs/core/entities/MRTextEntity */ \"./src/core/entities/MRTextEntity.js\");\n/* harmony import */ var _core_entities_MRTextAreaEntity__WEBPACK_IMPORTED_MODULE_23__ = __webpack_require__(/*! mrjs/core/entities/MRTextAreaEntity */ \"./src/core/entities/MRTextAreaEntity.js\");\n/* harmony import */ var _core_entities_MRTextFieldEntity__WEBPACK_IMPORTED_MODULE_24__ = __webpack_require__(/*! mrjs/core/entities/MRTextFieldEntity */ \"./src/core/entities/MRTextFieldEntity.js\");\n/* harmony import */ var _core_entities_MRTextInputEntity__WEBPACK_IMPORTED_MODULE_25__ = __webpack_require__(/*! ./core/entities/MRTextInputEntity */ \"./src/core/entities/MRTextInputEntity.js\");\n/* harmony import */ var _core_entities_MRVideoEntity__WEBPACK_IMPORTED_MODULE_26__ = __webpack_require__(/*! mrjs/core/entities/MRVideoEntity */ \"./src/core/entities/MRVideoEntity.js\");\n/* harmony import */ var _core_entities_MRVolumeEntity__WEBPACK_IMPORTED_MODULE_27__ = __webpack_require__(/*! mrjs/core/entities/MRVolumeEntity */ \"./src/core/entities/MRVolumeEntity.js\");\n/* harmony import */ var _core_componentSystems_AnchorSystem__WEBPACK_IMPORTED_MODULE_28__ = __webpack_require__(/*! ./core/componentSystems/AnchorSystem */ \"./src/core/componentSystems/AnchorSystem.js\");\n/* harmony import */ var _core_componentSystems_AnimationSystem__WEBPACK_IMPORTED_MODULE_29__ = __webpack_require__(/*! ./core/componentSystems/AnimationSystem */ \"./src/core/componentSystems/AnimationSystem.js\");\n/* harmony import */ var _core_componentSystems_BoundaryVisibilitySystem__WEBPACK_IMPORTED_MODULE_30__ = __webpack_require__(/*! ./core/componentSystems/BoundaryVisibilitySystem */ \"./src/core/componentSystems/BoundaryVisibilitySystem.js\");\n/* harmony import */ var _core_componentSystems_ClippingSystem__WEBPACK_IMPORTED_MODULE_31__ = __webpack_require__(/*! ./core/componentSystems/ClippingSystem */ \"./src/core/componentSystems/ClippingSystem.js\");\n/* harmony import */ var _core_componentSystems_ControlSystem__WEBPACK_IMPORTED_MODULE_32__ = __webpack_require__(/*! ./core/componentSystems/ControlSystem */ \"./src/core/componentSystems/ControlSystem.js\");\n/* harmony import */ var _core_componentSystems_GeometryStyleSystem__WEBPACK_IMPORTED_MODULE_33__ = __webpack_require__(/*! ./core/componentSystems/GeometryStyleSystem */ \"./src/core/componentSystems/GeometryStyleSystem.js\");\n/* harmony import */ var _core_componentSystems_InstancingSystem__WEBPACK_IMPORTED_MODULE_34__ = __webpack_require__(/*! ./core/componentSystems/InstancingSystem */ \"./src/core/componentSystems/InstancingSystem.js\");\n/* harmony import */ var _core_componentSystems_LayoutSystem__WEBPACK_IMPORTED_MODULE_35__ = __webpack_require__(/*! ./core/componentSystems/LayoutSystem */ \"./src/core/componentSystems/LayoutSystem.js\");\n/* harmony import */ var _core_componentSystems_MaskingSystem__WEBPACK_IMPORTED_MODULE_36__ = __webpack_require__(/*! ./core/componentSystems/MaskingSystem */ \"./src/core/componentSystems/MaskingSystem.js\");\n/* harmony import */ var _core_componentSystems_MaterialStyleSystem__WEBPACK_IMPORTED_MODULE_37__ = __webpack_require__(/*! ./core/componentSystems/MaterialStyleSystem */ \"./src/core/componentSystems/MaterialStyleSystem.js\");\n/* harmony import */ var _core_componentSystems_PhysicsSystem__WEBPACK_IMPORTED_MODULE_38__ = __webpack_require__(/*! ./core/componentSystems/PhysicsSystem */ \"./src/core/componentSystems/PhysicsSystem.js\");\n/* harmony import */ var _core_componentSystems_SkyBoxSystem__WEBPACK_IMPORTED_MODULE_39__ = __webpack_require__(/*! ./core/componentSystems/SkyBoxSystem */ \"./src/core/componentSystems/SkyBoxSystem.js\");\n/* harmony import */ var _core_componentSystems_StatsSystem__WEBPACK_IMPORTED_MODULE_40__ = __webpack_require__(/*! ./core/componentSystems/StatsSystem */ \"./src/core/componentSystems/StatsSystem.js\");\n/* harmony import */ var _core_componentSystems_TextSystem__WEBPACK_IMPORTED_MODULE_41__ = __webpack_require__(/*! ./core/componentSystems/TextSystem */ \"./src/core/componentSystems/TextSystem.js\");\n/* harmony import */ var _utils_index_js__WEBPACK_IMPORTED_MODULE_42__ = __webpack_require__(/*! ./utils/index.js */ \"./src/utils/index.js\");\n/* harmony import */ var _extras_index_js__WEBPACK_IMPORTED_MODULE_43__ = __webpack_require__(/*! ./extras/index.js */ \"./src/extras/index.js\");\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_44__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/**\n * @module mrjs\n * @description The one-stop-shop module for mrjs (including an import for mrjsUtils). Includes the ability to access threejs directly as well.\n */\n\n// TODO - this should auto grab instead of manually be updated as manual updates will create problems.\n// for import and export\n\n// Log the version number\n\nconsole.log('Current ᴍʀjs Version:', _package_json__WEBPACK_IMPORTED_MODULE_0__.version);\n\n// STYLE\n\n\n// GLOBAL\n\n\n// DATATYPES\n\n\n// DATAMANAGERS\n\n\n// CORE\n\n\n\n\n// CORE - USER\n\n\n// CORE - ENTITIES\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n// CORE - COMPONENT-SYSTEMS\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n// EXPORTS\n// UTILS\n\n// EXTRAS\n\n// THREE - So users dont need a separate versioning import for it.\n\n// MRJS - Exporting only necessary items for users to overwrite as they use MRjs.\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n//# sourceURL=webpack://mrjs/./src/index.js?"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ MRButtonEntity: () => (/* reexport safe */ _core_entities_MRButtonEntity__WEBPACK_IMPORTED_MODULE_12__.MRButtonEntity),\n/* harmony export */ MRDivEntity: () => (/* reexport safe */ _core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_13__.MRDivEntity),\n/* harmony export */ MREntity: () => (/* reexport safe */ _core_MREntity__WEBPACK_IMPORTED_MODULE_8__.MREntity),\n/* harmony export */ MRImageEntity: () => (/* reexport safe */ _core_entities_MRImageEntity__WEBPACK_IMPORTED_MODULE_15__.MRImageEntity),\n/* harmony export */ MRLightEntity: () => (/* reexport safe */ _core_entities_MRLightEntity__WEBPACK_IMPORTED_MODULE_16__.MRLightEntity),\n/* harmony export */ MRMediaEntity: () => (/* reexport safe */ _core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_17__.MRMediaEntity),\n/* harmony export */ MRModelEntity: () => (/* reexport safe */ _core_entities_MRModelEntity__WEBPACK_IMPORTED_MODULE_18__.MRModelEntity),\n/* harmony export */ MRPanelEntity: () => (/* reexport safe */ _core_entities_MRPanelEntity__WEBPACK_IMPORTED_MODULE_19__.MRPanelEntity),\n/* harmony export */ MRSkyBoxEntity: () => (/* reexport safe */ _core_entities_MRSkyBoxEntity__WEBPACK_IMPORTED_MODULE_20__.MRSkyBoxEntity),\n/* harmony export */ MRSystem: () => (/* reexport safe */ _core_MRSystem__WEBPACK_IMPORTED_MODULE_9__.MRSystem),\n/* harmony export */ MRTextAreaEntity: () => (/* reexport safe */ _core_entities_MRTextAreaEntity__WEBPACK_IMPORTED_MODULE_23__.MRTextAreaEntity),\n/* harmony export */ MRTextEntity: () => (/* reexport safe */ _core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_22__.MRTextEntity),\n/* harmony export */ MRTextFieldEntity: () => (/* reexport safe */ _core_entities_MRTextFieldEntity__WEBPACK_IMPORTED_MODULE_24__.MRTextFieldEntity),\n/* harmony export */ MRVideoEntity: () => (/* reexport safe */ _core_entities_MRVideoEntity__WEBPACK_IMPORTED_MODULE_26__.MRVideoEntity),\n/* harmony export */ MRVolumeEntity: () => (/* reexport safe */ _core_entities_MRVolumeEntity__WEBPACK_IMPORTED_MODULE_27__.MRVolumeEntity),\n/* harmony export */ Refractor: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.Refractor),\n/* harmony export */ THREE: () => (/* reexport module object */ three__WEBPACK_IMPORTED_MODULE_42__),\n/* harmony export */ Water: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.Water),\n/* harmony export */ WaterRefractionShader: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.WaterRefractionShader),\n/* harmony export */ WaterSystem: () => (/* reexport safe */ _extras_index_js__WEBPACK_IMPORTED_MODULE_43__.WaterSystem),\n/* harmony export */ mrjsUtils: () => (/* reexport safe */ _utils_index_js__WEBPACK_IMPORTED_MODULE_44__.mrjsUtils)\n/* harmony export */ });\n/* harmony import */ var _package_json__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../package.json */ \"./package.json\");\n/* harmony import */ var _package_json__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_package_json__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _defaultStyle_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./defaultStyle.css */ \"./src/defaultStyle.css\");\n/* harmony import */ var _global__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./global */ \"./src/global.js\");\n/* harmony import */ var _global__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(_global__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var _dataTypes_MRClippingGeometry__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./dataTypes/MRClippingGeometry */ \"./src/dataTypes/MRClippingGeometry.js\");\n/* harmony import */ var _dataTypes_MRPlane__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./dataTypes/MRPlane */ \"./src/dataTypes/MRPlane.js\");\n/* harmony import */ var _dataManagers_MRPlaneManager__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./dataManagers/MRPlaneManager */ \"./src/dataManagers/MRPlaneManager.js\");\n/* harmony import */ var _core_MRApp__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./core/MRApp */ \"./src/core/MRApp.js\");\n/* harmony import */ var _core_MRElement__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./core/MRElement */ \"./src/core/MRElement.js\");\n/* harmony import */ var _core_MREntity__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! mrjs/core/MREntity */ \"./src/core/MREntity.js\");\n/* harmony import */ var _core_MRSystem__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! mrjs/core/MRSystem */ \"./src/core/MRSystem.js\");\n/* harmony import */ var _core_user_MRHand__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ./core/user/MRHand */ \"./src/core/user/MRHand.js\");\n/* harmony import */ var _core_user_MRUser__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ./core/user/MRUser */ \"./src/core/user/MRUser.js\");\n/* harmony import */ var _core_entities_MRButtonEntity__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! mrjs/core/entities/MRButtonEntity */ \"./src/core/entities/MRButtonEntity.js\");\n/* harmony import */ var _core_entities_MRDivEntity__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! mrjs/core/entities/MRDivEntity */ \"./src/core/entities/MRDivEntity.js\");\n/* harmony import */ var _core_entities_MRHyperlinkEntity__WEBPACK_IMPORTED_MODULE_14__ = __webpack_require__(/*! mrjs/core/entities/MRHyperlinkEntity */ \"./src/core/entities/MRHyperlinkEntity.js\");\n/* harmony import */ var _core_entities_MRImageEntity__WEBPACK_IMPORTED_MODULE_15__ = __webpack_require__(/*! mrjs/core/entities/MRImageEntity */ \"./src/core/entities/MRImageEntity.js\");\n/* harmony import */ var _core_entities_MRLightEntity__WEBPACK_IMPORTED_MODULE_16__ = __webpack_require__(/*! mrjs/core/entities/MRLightEntity */ \"./src/core/entities/MRLightEntity.js\");\n/* harmony import */ var _core_entities_MRMediaEntity__WEBPACK_IMPORTED_MODULE_17__ = __webpack_require__(/*! mrjs/core/entities/MRMediaEntity */ \"./src/core/entities/MRMediaEntity.js\");\n/* harmony import */ var _core_entities_MRModelEntity__WEBPACK_IMPORTED_MODULE_18__ = __webpack_require__(/*! mrjs/core/entities/MRModelEntity */ \"./src/core/entities/MRModelEntity.js\");\n/* harmony import */ var _core_entities_MRPanelEntity__WEBPACK_IMPORTED_MODULE_19__ = __webpack_require__(/*! mrjs/core/entities/MRPanelEntity */ \"./src/core/entities/MRPanelEntity.js\");\n/* harmony import */ var _core_entities_MRSkyBoxEntity__WEBPACK_IMPORTED_MODULE_20__ = __webpack_require__(/*! mrjs/core/entities/MRSkyBoxEntity */ \"./src/core/entities/MRSkyBoxEntity.js\");\n/* harmony import */ var _core_entities_MRStatsEntity__WEBPACK_IMPORTED_MODULE_21__ = __webpack_require__(/*! ./core/entities/MRStatsEntity */ \"./src/core/entities/MRStatsEntity.js\");\n/* harmony import */ var _core_entities_MRTextEntity__WEBPACK_IMPORTED_MODULE_22__ = __webpack_require__(/*! mrjs/core/entities/MRTextEntity */ \"./src/core/entities/MRTextEntity.js\");\n/* harmony import */ var _core_entities_MRTextAreaEntity__WEBPACK_IMPORTED_MODULE_23__ = __webpack_require__(/*! mrjs/core/entities/MRTextAreaEntity */ \"./src/core/entities/MRTextAreaEntity.js\");\n/* harmony import */ var _core_entities_MRTextFieldEntity__WEBPACK_IMPORTED_MODULE_24__ = __webpack_require__(/*! mrjs/core/entities/MRTextFieldEntity */ \"./src/core/entities/MRTextFieldEntity.js\");\n/* harmony import */ var _core_entities_MRTextInputEntity__WEBPACK_IMPORTED_MODULE_25__ = __webpack_require__(/*! ./core/entities/MRTextInputEntity */ \"./src/core/entities/MRTextInputEntity.js\");\n/* harmony import */ var _core_entities_MRVideoEntity__WEBPACK_IMPORTED_MODULE_26__ = __webpack_require__(/*! mrjs/core/entities/MRVideoEntity */ \"./src/core/entities/MRVideoEntity.js\");\n/* harmony import */ var _core_entities_MRVolumeEntity__WEBPACK_IMPORTED_MODULE_27__ = __webpack_require__(/*! mrjs/core/entities/MRVolumeEntity */ \"./src/core/entities/MRVolumeEntity.js\");\n/* harmony import */ var _core_componentSystems_AnchorSystem__WEBPACK_IMPORTED_MODULE_28__ = __webpack_require__(/*! ./core/componentSystems/AnchorSystem */ \"./src/core/componentSystems/AnchorSystem.js\");\n/* harmony import */ var _core_componentSystems_AnimationSystem__WEBPACK_IMPORTED_MODULE_29__ = __webpack_require__(/*! ./core/componentSystems/AnimationSystem */ \"./src/core/componentSystems/AnimationSystem.js\");\n/* harmony import */ var _core_componentSystems_BoundaryVisibilitySystem__WEBPACK_IMPORTED_MODULE_30__ = __webpack_require__(/*! ./core/componentSystems/BoundaryVisibilitySystem */ \"./src/core/componentSystems/BoundaryVisibilitySystem.js\");\n/* harmony import */ var _core_componentSystems_ClippingSystem__WEBPACK_IMPORTED_MODULE_31__ = __webpack_require__(/*! ./core/componentSystems/ClippingSystem */ \"./src/core/componentSystems/ClippingSystem.js\");\n/* harmony import */ var _core_componentSystems_ControlSystem__WEBPACK_IMPORTED_MODULE_32__ = __webpack_require__(/*! ./core/componentSystems/ControlSystem */ \"./src/core/componentSystems/ControlSystem.js\");\n/* harmony import */ var _core_componentSystems_GeometryStyleSystem__WEBPACK_IMPORTED_MODULE_33__ = __webpack_require__(/*! ./core/componentSystems/GeometryStyleSystem */ \"./src/core/componentSystems/GeometryStyleSystem.js\");\n/* harmony import */ var _core_componentSystems_InstancingSystem__WEBPACK_IMPORTED_MODULE_34__ = __webpack_require__(/*! ./core/componentSystems/InstancingSystem */ \"./src/core/componentSystems/InstancingSystem.js\");\n/* harmony import */ var _core_componentSystems_LayoutSystem__WEBPACK_IMPORTED_MODULE_35__ = __webpack_require__(/*! ./core/componentSystems/LayoutSystem */ \"./src/core/componentSystems/LayoutSystem.js\");\n/* harmony import */ var _core_componentSystems_MaskingSystem__WEBPACK_IMPORTED_MODULE_36__ = __webpack_require__(/*! ./core/componentSystems/MaskingSystem */ \"./src/core/componentSystems/MaskingSystem.js\");\n/* harmony import */ var _core_componentSystems_MaterialStyleSystem__WEBPACK_IMPORTED_MODULE_37__ = __webpack_require__(/*! ./core/componentSystems/MaterialStyleSystem */ \"./src/core/componentSystems/MaterialStyleSystem.js\");\n/* harmony import */ var _core_componentSystems_PhysicsSystem__WEBPACK_IMPORTED_MODULE_38__ = __webpack_require__(/*! ./core/componentSystems/PhysicsSystem */ \"./src/core/componentSystems/PhysicsSystem.js\");\n/* harmony import */ var _core_componentSystems_SkyBoxSystem__WEBPACK_IMPORTED_MODULE_39__ = __webpack_require__(/*! ./core/componentSystems/SkyBoxSystem */ \"./src/core/componentSystems/SkyBoxSystem.js\");\n/* harmony import */ var _core_componentSystems_StatsSystem__WEBPACK_IMPORTED_MODULE_40__ = __webpack_require__(/*! ./core/componentSystems/StatsSystem */ \"./src/core/componentSystems/StatsSystem.js\");\n/* harmony import */ var _core_componentSystems_TextSystem__WEBPACK_IMPORTED_MODULE_41__ = __webpack_require__(/*! ./core/componentSystems/TextSystem */ \"./src/core/componentSystems/TextSystem.js\");\n/* harmony import */ var three__WEBPACK_IMPORTED_MODULE_42__ = __webpack_require__(/*! three */ \"./node_modules/three/build/three.module.js\");\n/* harmony import */ var _extras_index_js__WEBPACK_IMPORTED_MODULE_43__ = __webpack_require__(/*! ./extras/index.js */ \"./src/extras/index.js\");\n/* harmony import */ var _utils_index_js__WEBPACK_IMPORTED_MODULE_44__ = __webpack_require__(/*! ./utils/index.js */ \"./src/utils/index.js\");\n/**\n * @module mrjs\n * @description The one-stop-shop module for mrjs (including an import for mrjsUtils). Includes the ability to access threejs directly as well.\n */\n\n// TODO - this should auto grab instead of manually be updated as manual updates will create problems.\n// for import and export\n\n// Log the version number\n\nconsole.log('Current ᴍʀjs Version:', _package_json__WEBPACK_IMPORTED_MODULE_0__.version);\n\n// STYLE\n\n\n// GLOBAL\n\n\n// DATATYPES\n\n\n// DATAMANAGERS\n\n\n// CORE\n\n\n\n\n// CORE - USER\n\n\n// CORE - ENTITIES\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n// CORE - COMPONENT-SYSTEMS\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n// EXPORTS\n\n// THREE - So users dont need a separate versioning import for it.\n\n\n// MRJS - Exporting only necessary items for users to overwrite as they use MRjs.\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n// EXTRAS\n\n\n// UTILS - exporting as a named group since it's a submodule of this js module\n\n\n\n//# sourceURL=webpack://mrjs/./src/index.js?"); /***/ }), diff --git a/package.json b/package.json index ba2271cd..6feaad6b 100644 --- a/package.json +++ b/package.json @@ -17,6 +17,7 @@ "test": "NODE_OPTIONS=--experimental-vm-modules npx jest", "test-serially": "NODE_OPTIONS=--experimental-vm-modules npx jest --runInBand", "test-randomized": "NODE_OPTIONS=--experimental-vm-modules npx jest --runInBand --random", + "clear-testing-cache": "npx jest --clearCache", "docs": "./scripts/create-docs.sh", "prettier-check": "prettier --check \"src/**/*.js\" \"*.js\"", "prettier-fix": "prettier --write \"src/**/*.js\" \"*.js\"", diff --git a/samples/examples/skybox.html b/samples/examples/skybox.html index 702e129a..8ab490ee 100644 --- a/samples/examples/skybox.html +++ b/samples/examples/skybox.html @@ -5,6 +5,7 @@