diff --git a/doc/image_orientations.gif b/doc/image_orientations.gif new file mode 100755 index 0000000000000000000000000000000000000000..89c4c3ea1444de6b32e1cecc2d2825e430c833e8 Binary files /dev/null and b/doc/image_orientations.gif differ diff --git a/src/assets/hud/spawn_photo-hover.png b/src/assets/hud/spawn_photo-hover.png new file mode 100755 index 0000000000000000000000000000000000000000..178e75cf5d2ca06e39885eccd93475f96f3175f5 Binary files /dev/null and b/src/assets/hud/spawn_photo-hover.png differ diff --git a/src/assets/hud/spawn_photo.png b/src/assets/hud/spawn_photo.png new file mode 100755 index 0000000000000000000000000000000000000000..00473bb8faa20e958effb45ee17b651d730510dc Binary files /dev/null and b/src/assets/hud/spawn_photo.png differ diff --git a/src/assets/stylesheets/2d-hud.scss b/src/assets/stylesheets/2d-hud.scss index edc8b391990a9c81dd67ca2579ffc2991a169ac3..67fbc0ada8db77ad2e2fee42b62ebb1f1c00f624 100644 --- a/src/assets/stylesheets/2d-hud.scss +++ b/src/assets/stylesheets/2d-hud.scss @@ -5,19 +5,28 @@ display: flex; justify-content: center; align-items: center; - height: 80px; width: 100%; user-select: none; &:local(.top) { top: 10px; + height: 80px; } - &:local(.bottom) { + &:local(.column) { + flex-direction: column; bottom: 20px; } } +:local(.bottom) { + margin-bottom: 20px; +} + +:local(.hide) { + display: none; +} + :local(.panel) { display: flex; justify-content: space-around; @@ -42,6 +51,14 @@ margin-left: -40px; } +:local(.panel.up) { + border-top-right-radius: 30px; + border-top-left-radius: 30px; + padding-top: 5px; + padding-bottom: 45px; + margin-bottom: -40px; +} + :local(.iconButton) { width: 40px; height: 40px; @@ -109,3 +126,7 @@ :local(.iconButton.create-object:hover) { background-image: url(../hud/create_object-hover.png); } + +:local(.iconButton.mobile-media-picker) { + background-image: url(../hud/spawn_photo.png); +} diff --git a/src/components/offset-relative-to.js b/src/components/offset-relative-to.js index e00acd4e57a1842e5b8116c5448a3c2088443eab..877cfcf8a118a13eb94200925d1ac53053d9faa1 100644 --- a/src/components/offset-relative-to.js +++ b/src/components/offset-relative-to.js @@ -13,6 +13,9 @@ AFRAME.registerComponent("offset-relative-to", { on: { type: "string" }, + orientation: { + default: 1 // see doc/image_orientations.gif + }, selfDestruct: { default: false } @@ -27,6 +30,9 @@ AFRAME.registerComponent("offset-relative-to", { }, updateOffset: (function() { + const y = new THREE.Vector3(0, 1, 0); + const z = new THREE.Vector3(0, 0, -1); + const QUARTER_CIRCLE = Math.PI / 2; const offsetVector = new THREE.Vector3(); return function() { const obj = this.el.object3D; @@ -40,6 +46,38 @@ AFRAME.registerComponent("offset-relative-to", { this.el.body && this.el.body.position.copy(obj.position); target.getWorldQuaternion(obj.quaternion); this.el.body && this.el.body.quaternion.copy(obj.quaternion); + + // See doc/image_orientations.gif + switch (this.data.orientation) { + case 8: + obj.rotateOnAxis(z, 3 * QUARTER_CIRCLE); + break; + case 7: + obj.rotateOnAxis(z, 3 * QUARTER_CIRCLE); + obj.rotateOnAxis(y, 2 * QUARTER_CIRCLE); + break; + case 6: + obj.rotateOnAxis(z, QUARTER_CIRCLE); + break; + case 5: + obj.rotateOnAxis(z, QUARTER_CIRCLE); + obj.rotateOnAxis(y, 2 * QUARTER_CIRCLE); + break; + case 4: + obj.rotateOnAxis(z, 2 * QUARTER_CIRCLE); + obj.rotateOnAxis(y, 2 * QUARTER_CIRCLE); + break; + case 3: + obj.rotateOnAxis(z, 2 * QUARTER_CIRCLE); + break; + case 2: + obj.rotateOnAxis(y, 2 * QUARTER_CIRCLE); + break; + case 1: + default: + break; + } + if (this.data.selfDestruct) { if (this.data.on) { this.el.sceneEl.removeEventListener(this.data.on, this.updateOffset); diff --git a/src/components/super-spawner.js b/src/components/super-spawner.js index f760a29e25cc1fb680b55a0b504a42202948f0da..0c69e82da5cfe8c62e4aabafd2506fcf11613e3d 100644 --- a/src/components/super-spawner.js +++ b/src/components/super-spawner.js @@ -84,7 +84,7 @@ AFRAME.registerComponent("super-spawner", { const thisGrabId = nextGrabId++; this.heldEntities.set(hand, thisGrabId); - const entity = addMedia(this.data.src, ObjectContentOrigins.SPAWNER); + const entity = addMedia(this.data.src, ObjectContentOrigins.SPAWNER).entity; entity.object3D.position.copy( this.data.useCustomSpawnPosition ? this.data.spawnPosition : this.el.object3D.position ); diff --git a/src/components/virtual-gamepad-controls.css b/src/components/virtual-gamepad-controls.css index 4270c36ad7be261997f7ab77218e9ea817269620..3a5ed3c9c6aeb7ebda7a3b8f4cee610af2a78a57 100644 --- a/src/components/virtual-gamepad-controls.css +++ b/src/components/virtual-gamepad-controls.css @@ -6,11 +6,11 @@ :local(.touchZone.left) { left: 0; - right: 50%; + right: 55%; } :local(.touchZone.right) { - left: 50%; + left: 55%; right: 0; } diff --git a/src/hub.js b/src/hub.js index 8db01eaddef8c92fa5dd4a8da040fa2473615b56..5e347527cfd99e822cb0d6adf6c9cb3d0c7bca77 100644 --- a/src/hub.js +++ b/src/hub.js @@ -306,11 +306,14 @@ const onReady = async () => { const offset = { x: 0, y: 0, z: -1.5 }; const spawnMediaInfrontOfPlayer = (src, contentOrigin) => { - const entity = addMedia(src, contentOrigin, true); + const { entity, orientation } = addMedia(src, contentOrigin, true); - entity.setAttribute("offset-relative-to", { - target: "#player-camera", - offset + orientation.then(or => { + entity.setAttribute("offset-relative-to", { + target: "#player-camera", + offset, + orientation: or + }); }); }; diff --git a/src/react-components/2d-hud.js b/src/react-components/2d-hud.js index 15168e999bd4576f80b0f4242ed60675f4557699..ee5d75df591373e1d2b1bb6fdb2112cf3993affd 100644 --- a/src/react-components/2d-hud.js +++ b/src/react-components/2d-hud.js @@ -37,18 +37,43 @@ TopHUD.propTypes = { onToggleSpaceBubble: PropTypes.func }; -const BottomHUD = ({ onCreateObject }) => ( - <div className={cx(styles.container, styles.bottom)}> - <div - className={cx("ui-interactive", styles.iconButton, styles.large, styles.createObject)} - title={"Create Object"} - onClick={onCreateObject} - /> +const BottomHUD = ({ onCreateObject, showPhotoPicker, onMediaPicked }) => ( + <div className={cx(styles.container, styles.column, styles.bottom)}> + {showPhotoPicker ? ( + <div className={cx("ui-interactive", styles.panel, styles.up)}> + <input + id="media-picker-input" + className={cx(styles.hide)} + type="file" + accept="image/*" + multiple + onChange={e => { + for (const file of e.target.files) { + onMediaPicked(file); + } + }} + /> + <label htmlFor="media-picker-input"> + <div className={cx(styles.iconButton, styles.mobileMediaPicker)} title={"Pick Media"} /> + </label> + </div> + ) : ( + <div /> + )} + <div> + <div + className={cx("ui-interactive", styles.iconButton, styles.large, styles.createObject)} + title={"Create Object"} + onClick={onCreateObject} + /> + </div> </div> ); BottomHUD.propTypes = { - onCreateObject: PropTypes.func + onCreateObject: PropTypes.func, + showPhotoPicker: PropTypes.bool, + onMediaPicked: PropTypes.func }; export default { TopHUD, BottomHUD }; diff --git a/src/react-components/ui-root.js b/src/react-components/ui-root.js index 36b5434714a0bca8af2fc7dc4f2f38d094952f3e..4c2a9ffdfda1af43ee6848b7955fc6752684e3fe 100644 --- a/src/react-components/ui-root.js +++ b/src/react-components/ui-root.js @@ -526,8 +526,8 @@ class UIRoot extends Component { this.setState({ infoDialogType: null, linkCode: null, linkCodeCancel: null }); }; - handleCreateObject = url => { - this.props.scene.emit("add_media", url); + handleCreateObject = media => { + this.props.scene.emit("add_media", media); }; render() { @@ -910,6 +910,8 @@ class UIRoot extends Component { )} <TwoDHUD.BottomHUD onCreateObject={() => this.setState({ infoDialogType: InfoDialog.dialogTypes.create_object })} + showPhotoPicker={AFRAME.utils.device.isMobile()} + onMediaPicked={this.handleCreateObject} /> </div> ) : null} diff --git a/src/utils/media-utils.js b/src/utils/media-utils.js index 83a5f788ea841c1f3343c2068293b97bf9f53f3e..825865dcac289bc2afa1f8f3ed878b485e682ee1 100644 --- a/src/utils/media-utils.js +++ b/src/utils/media-utils.js @@ -51,6 +51,45 @@ export const upload = file => { }).then(r => r.json()); }; +// https://stackoverflow.com/questions/7584794/accessing-jpeg-exif-rotation-data-in-javascript-on-the-client-side/32490603#32490603 +function getOrientation(file, callback) { + const reader = new FileReader(); + reader.onload = function(e) { + const view = new DataView(e.target.result); + if (view.getUint16(0, false) != 0xffd8) { + return callback(-2); + } + const length = view.byteLength; + let offset = 2; + while (offset < length) { + if (view.getUint16(offset + 2, false) <= 8) return callback(-1); + const marker = view.getUint16(offset, false); + offset += 2; + if (marker == 0xffe1) { + if (view.getUint32((offset += 2), false) != 0x45786966) { + return callback(-1); + } + + const little = view.getUint16((offset += 6), false) == 0x4949; + offset += view.getUint32(offset + 4, little); + const tags = view.getUint16(offset, little); + offset += 2; + for (let i = 0; i < tags; i++) { + if (view.getUint16(offset + i * 12, little) == 0x0112) { + return callback(view.getUint16(offset + i * 12 + 8, little)); + } + } + } else if ((marker & 0xff00) != 0xff00) { + break; + } else { + offset += view.getUint16(offset, false); + } + } + return callback(-1); + }; + reader.readAsArrayBuffer(file); +} + let interactableId = 0; export const addMedia = (src, contentOrigin, resize = false) => { const scene = AFRAME.scenes[0]; @@ -61,6 +100,15 @@ export const addMedia = (src, contentOrigin, resize = false) => { entity.setAttribute("media-loader", { resize, src: typeof src === "string" ? src : "" }); scene.appendChild(entity); + const orientation = new Promise(function(resolve) { + if (src instanceof File) { + getOrientation(src, x => { + resolve(x); + }); + } else { + resolve(1); + } + }); if (src instanceof File) { upload(src) .then(response => { @@ -78,5 +126,5 @@ export const addMedia = (src, contentOrigin, resize = false) => { scene.emit("object_spawned", { objectType }); }); - return entity; + return { entity, orientation }; };