parent
ae98cf9ca0
commit
c5f1f066dd
|
@ -0,0 +1,84 @@
|
|||
/*
|
||||
*Usage:
|
||||
* <a-entity controlattach="el: #leftHand; class: foo"></a-entity>
|
||||
* <a-entity id="leftHand">
|
||||
* <a-entity class="foo" position="0 0 0" rotation="0 0 0" scale="0 0 0"></a-entity>
|
||||
* <a-entity>
|
||||
*
|
||||
* NOTE: you can hint different position/rotation-offsets by adding controller-specific entities to the target:
|
||||
*
|
||||
* <a-entity id="leftHand">
|
||||
* <a-entity class="foo" position="0 0 0" rotation="0 0 0" scale="0 0 0"></a-entity>
|
||||
* <a-entity class="foo hand-tracking-controls" position="0 0 0" rotation="0 0 0" scale="0 0 0"></a-entity>
|
||||
* <a-entity>
|
||||
*
|
||||
* The controllernames are hinted by the 'controllermodelready' event:
|
||||
*
|
||||
* 'hand-tracking-controls',
|
||||
* 'meta-touch-controls',
|
||||
* 'valve-index-controls',
|
||||
* 'logitech-mx-ink-controls',
|
||||
* 'windows-motion-controls',
|
||||
* 'hp-mixed-reality-controls',
|
||||
* 'generic-tracked-controller-controls',
|
||||
* 'pico-controls',
|
||||
* (and more in the future)
|
||||
*/
|
||||
|
||||
AFRAME.registerComponent('controlattach', {
|
||||
schema:{
|
||||
el: {type:"selector"},
|
||||
class: {type:"string"}
|
||||
},
|
||||
init: function(){
|
||||
if( !this.data.el ) return console.warn(`controlattach.js: cannot find ${this.data.el}`)
|
||||
this.controllers = {}
|
||||
this.remember()
|
||||
this.data.el.addEventListener('controllermodelready', this.bindPlaceHolders.bind(this,["controllermodelready"]) )
|
||||
this.data.el.addEventListener('controllerconnected', this.bindPlaceHolders.bind(this,["controllerconnected"] ) )
|
||||
},
|
||||
|
||||
bindPlaceHolders: function(type,e){
|
||||
let controllerName = e.detail.name
|
||||
let selector = `.${this.data.class}.${controllerName}`
|
||||
let placeholder = this.data.el.querySelector(selector)
|
||||
if( !placeholder ){
|
||||
placeholder = this.data.el.querySelector(`.${this.data.class}`)
|
||||
console.log("fallback")
|
||||
}
|
||||
if( !placeholder ) return console.warn("controlattach.js: could not find placeholder to attach to")
|
||||
this.attachPlaceHolder(type,e,placeholder, controllerName)
|
||||
},
|
||||
|
||||
attachPlaceHolder: function(type, el, placeholder, controllerName){
|
||||
this.el.object3DMap = {} // unsync THREE <-> AFRAME entity
|
||||
placeholder.object3D.add( this.obj )
|
||||
if( controllerName != 'hand-tracking-controls' ){
|
||||
// re-add for controller-models which don't re-add children ('meta-touch-controls' e.g.)
|
||||
if( this.data.el.getObject3D("mesh") ){
|
||||
this.data.el.getObject3D("mesh").add(placeholder.object3D)
|
||||
}
|
||||
}
|
||||
// these are handled by the placeholder entity
|
||||
this.obj.position.set(0,0,0);
|
||||
this.obj.rotation.set(0,0,0);
|
||||
this.obj.scale.set(1,1,1);
|
||||
},
|
||||
|
||||
detach: function(){
|
||||
this.el.setObject3D( this.obj.uuid, this.obj )
|
||||
this.el.object3D.position.copy( this.position )
|
||||
this.el.object3D.rotation.copy( this.rotation )
|
||||
this.el.object3D.scale.copy( this.scale )
|
||||
},
|
||||
|
||||
remember: function(){
|
||||
this.obj = this.el.object3D
|
||||
this.position = this.el.object3D.position.clone()
|
||||
this.rotation = this.el.object3D.rotation.clone()
|
||||
this.scale = this.el.object3D.scale.clone()
|
||||
this.parent = this.el.object3D.parent
|
||||
}
|
||||
|
||||
})
|
||||
|
|
@ -383,18 +383,13 @@ if( typeof AFRAME != 'undefined '){
|
|||
instance.addEventListener('window.onresize', resize )
|
||||
instance.addEventListener('window.onmaximize', resize )
|
||||
|
||||
const focus = (showdom) => (e) => {
|
||||
const focus = (e) => {
|
||||
this.el.emit('focus',e.detail)
|
||||
if( this.el.components.window && this.data.renderer == 'canvas'){
|
||||
this.el.components.window.show( showdom )
|
||||
}
|
||||
}
|
||||
|
||||
this.el.addEventListener('obbcollisionstarted', focus(false) )
|
||||
this.el.sceneEl.addEventListener('enter-vr', focus(false) )
|
||||
this.el.sceneEl.addEventListener('enter-ar', focus(false) )
|
||||
this.el.sceneEl.addEventListener('exit-vr', focus(true) )
|
||||
this.el.sceneEl.addEventListener('exit-ar', focus(true) )
|
||||
this.el.addEventListener('obbcollisionstarted', focus )
|
||||
this.el.sceneEl.addEventListener('exit-vr', focus )
|
||||
this.el.sceneEl.addEventListener('exit-ar', focus )
|
||||
|
||||
instance.object3D.quaternion.copy( AFRAME.scenes[0].camera.quaternion ) // face towards camera
|
||||
},
|
||||
|
|
|
@ -4,7 +4,7 @@
|
|||
* displays app (icons) in 2D and 3D handmenu (enduser can launch desktop-like 'apps')
|
||||
*
|
||||
* ```html
|
||||
* <a-entity launcher="attach: #left-hand"></a-entity>
|
||||
* <a-entity launcher="attach: #left-hand" wearable></a-entity>
|
||||
*
|
||||
* <a-assets>
|
||||
* <a-mixin id="menuitem" geometry="primitive: plane; width: 0.15; height: 0.15; depth: 0.02" obb-collider="size: 0.03 0.03 0.03" ></a-mixin>
|
||||
|
@ -90,28 +90,8 @@ AFRAME.registerComponent('launcher', {
|
|||
this.worldPosition = new THREE.Vector3()
|
||||
|
||||
this.el.setAttribute("dom","")
|
||||
this.el.setAttribute("pressable","")
|
||||
this.render()
|
||||
|
||||
// this.tick = AFRAME.utils.throttleTick( this.tick, 100, this );
|
||||
|
||||
// this.el.sceneEl.addEventListener('enter-vr', (e) => this.preventAccidentalButtonPresses() )
|
||||
// this.el.sceneEl.addEventListener('enter-ar', (e) => this.preventAccidentalButtonPresses() )
|
||||
// this.el.sceneEl.addEventListener('exit-vr', (e) => this.preventAccidentalButtonPresses() )
|
||||
// this.el.sceneEl.addEventListener('exit-ar', (e) => this.preventAccidentalButtonPresses() )
|
||||
|
||||
//if( this.data.attach ){
|
||||
// if( this.isHand(this.data.attach) ){
|
||||
// this.data.attach.addEventListener('model-loaded', () => {
|
||||
// this.ready = true
|
||||
// //this.data.attach.appendChild(this.el)
|
||||
// let armature = this.data.attach.object3D.getObjectByName('Armature')
|
||||
// if( !armature ) return console.warn('cannot find armature')
|
||||
// let object3D = this.el.object3D.children[0]
|
||||
// this.el.remove()
|
||||
// setTimeout( () => this.data.attach.object3D.add(object3D), 500)
|
||||
// })
|
||||
// }else this.data.attach.appendChild(this.el)
|
||||
//}else console.warn("launcher.js: attach-option not given")
|
||||
},
|
||||
|
||||
isHand: (el) => {
|
||||
|
@ -119,7 +99,7 @@ AFRAME.registerComponent('launcher', {
|
|||
},
|
||||
|
||||
dom: {
|
||||
scale: 1,
|
||||
scale: 0.4,
|
||||
events: ['click'],
|
||||
html: (me) => `<div class="iconmenu">loading components..</div>`,
|
||||
css: (me) => `.iconmenu {
|
||||
|
@ -203,14 +183,14 @@ AFRAME.registerComponent('launcher', {
|
|||
let i = 0
|
||||
let j = 0
|
||||
let colors = this.data.colors
|
||||
const add2D = (launchCom,el,manifest) => {
|
||||
const add2D = (launchCom,manifest) => {
|
||||
let btn = document.createElement('button')
|
||||
let iconDefault = ""
|
||||
btn.innerHTML = `${ manifest?.icons?.length > 0
|
||||
? `<img src='${manifest.icons[0].src || iconDefault}' title='${manifest.name}: ${manifest.description}'/>`
|
||||
: `${manifest.short_name || manifest.name}`
|
||||
}`
|
||||
btn.addEventListener('click', () => el.emit('launcher',{}) )
|
||||
btn.addEventListener('click', () => launchCom.launcher() )
|
||||
this.el.dom.appendChild(btn)
|
||||
}
|
||||
|
||||
|
@ -221,15 +201,21 @@ AFRAME.registerComponent('launcher', {
|
|||
// console.warn(`could not find component '${launchComponentKey}' (forgot to include script-tag?)`)
|
||||
const manifest = c.manifest
|
||||
if( manifest ){
|
||||
add2D(c,c.el,manifest)
|
||||
add2D(c,manifest)
|
||||
}
|
||||
})
|
||||
|
||||
this.setOriginToMiddle(this.el.object3D)
|
||||
},
|
||||
|
||||
tick: function(){
|
||||
setOriginToMiddle: function(object) {
|
||||
var box = new THREE.Box3().setFromObject(object);
|
||||
var center = new THREE.Vector3();
|
||||
box.getCenter(center);
|
||||
object.position.sub(center);
|
||||
},
|
||||
|
||||
|
||||
manifest: { // HTML5 manifest to identify app to xrsh
|
||||
"short_name": "launcher",
|
||||
"name": "App Launcher",
|
||||
|
@ -302,7 +288,11 @@ AFRAME.registerSystem('launcher',{
|
|||
name:"foo"+i,
|
||||
// icon: "https://..."
|
||||
description: "lorem ipsum",
|
||||
cb: () => alert("foo")
|
||||
cb: () => {
|
||||
let el = document.createElement("a-box")
|
||||
el.setAttribute("position", `${Math.random()*10} ${Math.random()*10} ${Math.random()*10}`)
|
||||
this.el.sceneEl.appendChild(el)
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
|
@ -330,7 +320,14 @@ AFRAME.registerSystem('launcher',{
|
|||
let searchEvent = 'launcher'
|
||||
let els = [...this.sceneEl.getElementsByTagName("*")]
|
||||
let seen = {}
|
||||
this.launchables = [];
|
||||
this.launchables = [
|
||||
/*
|
||||
* {
|
||||
* manifest: {...}
|
||||
* launcher: () => ....
|
||||
* }
|
||||
*/
|
||||
];
|
||||
|
||||
// collect manually registered launchables
|
||||
this.registered.map( (launchable) => this.launchables.push(launchable) )
|
||||
|
@ -341,7 +338,9 @@ AFRAME.registerSystem('launcher',{
|
|||
if( el.components ){
|
||||
for( let i in el.components ){
|
||||
if( el.components[i].events && el.components[i].events[searchEvent] && !seen[i] ){
|
||||
this.launchables.push(hasEvent = seen[i] = el.components[i])
|
||||
let com = hasEvent = seen[i] = el.components[i]
|
||||
com.launcher = () => com.el.emit('launcher',{})
|
||||
this.launchables.push(com)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -17,7 +17,9 @@ window.AFRAME.registerComponent('pinch-to-teleport', {
|
|||
pos.x += direction.x
|
||||
pos.z += direction.z
|
||||
// set the new position
|
||||
this.data.rig.setAttribute("position", pos);
|
||||
if( !this.data.rig ){
|
||||
this.data.rig.setAttribute("position", pos);
|
||||
}
|
||||
// !!! NOTE - it would be more efficient to do the
|
||||
// position change on the players THREE.Object:
|
||||
// `player.object3D.position.add(direction)`
|
||||
|
|
|
@ -2,13 +2,11 @@
|
|||
|
||||
AFRAME.registerComponent('pressable', {
|
||||
schema: {
|
||||
pressDistance: {
|
||||
default: 0.01
|
||||
}
|
||||
pressDistance: { default: 0.003 },
|
||||
pressDuration: { default: 300 }
|
||||
},
|
||||
init: function() {
|
||||
this.worldPosition = new THREE.Vector3();
|
||||
this.fingerWorldPosition = new THREE.Vector3();
|
||||
this.raycaster = new THREE.Raycaster()
|
||||
this.handEls = document.querySelectorAll('[hand-tracking-controls]');
|
||||
this.pressed = false;
|
||||
|
@ -34,39 +32,41 @@ AFRAME.registerComponent('pressable', {
|
|||
var handEl;
|
||||
let minDistance = 5
|
||||
|
||||
// compensate for xrf-get AFRAME component (which references non-reparented buffergeometries from the 3D model)
|
||||
let object3D = this.el.object3D.child || this.el.object3D
|
||||
// compensate for an object inside a group
|
||||
let object3D = this.el.object3D.type == "Group" ? this.el.object3D.children[0] : this.el.object3D
|
||||
|
||||
for (var i = 0; i < handEls.length; i++) {
|
||||
handEl = handEls[i];
|
||||
let indexTipPosition = handEl.components['hand-tracking-controls'].indexTipPosition
|
||||
// Apply the relative position to the parent's world position
|
||||
handEl.object3D.updateMatrixWorld();
|
||||
handEl.object3D.getWorldPosition( this.fingerWorldPosition )
|
||||
this.fingerWorldPosition.add( indexTipPosition )
|
||||
let indexTip = handEl.object3D.getObjectByName('index-finger-tip')
|
||||
if( ! indexTip ) return // nothing to do here
|
||||
|
||||
this.raycaster.far = this.data.pressDistance
|
||||
// Create a direction vector (doesnt matter because it is supershort for 'touch' purposes)
|
||||
const direction = new THREE.Vector3(1.0,0,0);
|
||||
this.raycaster.set(this.fingerWorldPosition, direction)
|
||||
|
||||
// Create a direction vector to negative Z
|
||||
const direction = new THREE.Vector3(0,0,-1.0);
|
||||
direction.normalize()
|
||||
this.raycaster.set(indexTip.position, direction)
|
||||
intersects = this.raycaster.intersectObjects([object3D],true)
|
||||
|
||||
object3D.getWorldPosition(this.worldPosition)
|
||||
|
||||
distance = this.fingerWorldPosition.distanceTo(this.worldPosition)
|
||||
distance = indexTip.position.distanceTo(this.worldPosition)
|
||||
minDistance = distance < minDistance ? distance : minDistance
|
||||
|
||||
if (intersects.length ){
|
||||
this.i = this.i || 0;
|
||||
if( !this.pressed ){
|
||||
this.el.emit('pressedstarted');
|
||||
this.el.emit('click');
|
||||
this.el.emit('pressedstarted', intersects);
|
||||
this.el.emit('click', intersects);
|
||||
document.querySelector('[isoterminal]').components.isoterminal.term.term.write("\r\nclick"+(++this.i))
|
||||
this.pressed = setTimeout( () => {
|
||||
this.el.emit('pressedended');
|
||||
this.el.emit('pressedended', intersects);
|
||||
this.pressed = null
|
||||
},300)
|
||||
}, this.data.pressDuration )
|
||||
}
|
||||
}
|
||||
}
|
||||
this.distance = minDistance
|
||||
}
|
||||
},
|
||||
|
||||
});
|
||||
|
|
|
@ -1,30 +1,56 @@
|
|||
AFRAME.registerComponent('wearable', {
|
||||
schema:{
|
||||
el: {type:"selector"},
|
||||
position: {type:"vec3"},
|
||||
rotation: {type:"vec3"}
|
||||
controlPos: {type:"vec3"},
|
||||
controlRot: {type:"vec3"},
|
||||
handPos: {type:"vec3"},
|
||||
handRot: {type:"vec3"}
|
||||
},
|
||||
init: function(){
|
||||
this.position = this.el.object3D.position.clone()
|
||||
this.rotation = this.el.object3D.rotation.clone()
|
||||
if( !this.el ) return console.warn(`wear.js: cannot find ${this.data.el}`)
|
||||
let ctl = this.data.el
|
||||
this.remember()
|
||||
|
||||
// hand vs controller attach-heuristics
|
||||
this.el.sceneEl.addEventListener('controllersupdated', (e) => {
|
||||
if( !this.data.el.components['hand-tracking-controls'].controllerPresent ){
|
||||
this.attach( ctl.components['hand-tracking-controls'].wristObject3D )
|
||||
}else{
|
||||
this.attach( ctl.object3D )
|
||||
}
|
||||
})
|
||||
if( !this.data.el ) return console.warn(`wear.js: cannot find ${this.data.el}`)
|
||||
this.data.el.object3D.name = 'wearable'
|
||||
|
||||
// hand vs controller multi attach-heuristics (intended to survived AFRAME updates)
|
||||
this.data.el.addEventListener('controllermodelready', this.attachWhatEverController.bind(this) ) // downside: no model yet
|
||||
this.data.el.addEventListener('model-loaded', this.attachWhatEverController.bind(this) ) // downside: only called once [model not added yet]
|
||||
this.el.sceneEl.addEventListener('controllersupdated', this.attachWhatEverController.bind(this) ) // downside: only called when switching [no model yet]
|
||||
},
|
||||
|
||||
attachWhatEverController: function(e){
|
||||
setTimeout( () => { // needed because the events are called before the model was added via add())
|
||||
let wrist = false
|
||||
let hand = this.data.el.components['hand-tracking-controls']
|
||||
if( hand && hand.controllerPresent ) wrist = hand.wristObject3D
|
||||
this.attach( wrist || this.data.el.object3D)
|
||||
this.update( wrist ? 'hand' : 'control')
|
||||
},100)
|
||||
},
|
||||
|
||||
attach: function(target){
|
||||
if( target.uuid == this.el.object3D.parent.uuid ) return; // already attached
|
||||
target.add(this.el.object3D)
|
||||
this.el.object3D.position.copy( this.data.position )
|
||||
this.el.object3D.rotation.copy( this.data.rotation )
|
||||
target.updateMatrixWorld();
|
||||
if( this.target && target.uuid == this.target.uuid ) return// already attached
|
||||
target.add(this.el.object3D )
|
||||
this.target = target
|
||||
},
|
||||
|
||||
detach: function(){
|
||||
this.parent.add(this.el.object3D)
|
||||
this.el.object3D.position.copy( this.position )
|
||||
this.el.object3D.rotation.copy( this.rotation )
|
||||
},
|
||||
|
||||
remember: function(){
|
||||
this.position = this.el.object3D.position.clone()
|
||||
this.rotation = this.el.object3D.rotation.clone()
|
||||
this.parent = this.el.object3D.parent
|
||||
},
|
||||
|
||||
update: function(type){
|
||||
let position = type == 'hand' ? this.data.handPos : this.data.controlPos
|
||||
let rotation = type == 'hand' ? this.data.handRot : this.data.controlRot
|
||||
this.el.object3D.position.copy( position )
|
||||
this.el.object3D.rotation.set( rotation.x, rotation.y, rotation.z )
|
||||
}
|
||||
|
||||
})
|
||||
|
|
Loading…
Reference in New Issue