18
13

More than 5 years have passed since last update.

A-FrameでOculus Questのコントローラを使う

Last updated at Posted at 2019-06-21

A-Frameで手っ取り早くOculus Questの入力を取得するサンプルです。
本記事では、getElementByIdで各コントローラの情報を取得したあとに入力イベントを追加する方法(sample1)と、AFRAME.registerComponentを使って入力情報を取得するコンポーネントを自作する方法(sample2)の2通りを掲載しています。どちらも同じような結果が得られますので、わかりやすい方を参考にしてください。

なおsample1のコードの概要はこちらのブログ[LINK]にて解説しています。
また、公式サイト[LINK]でsampl2で用いているcomponent追加方法の詳細が記載されています。
詳しいことを知りたい方ははぜひこれらをご一読ください。

1. 体験

Oculus Questを装着しWebブラウザでご覧ください。
https://quest-test.glitch.me

Videoデモ

2. サンプルコード

sample1

getElementByIdでコントローラを取得してアクションリスナを追加。

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
    <title>Oculus Quest Input</title>
    <meta name="description" content="Oculus Quest Input">
    <script src="https://aframe.io/releases/0.9.2/aframe.min.js"></script>
  </head>
  <body>
    <a-scene background="color: #FAFAFA">
      <a-box position="-1 0.5 -3" rotation="0 45 0" color="#4CC3D9" shadow></a-box>
      <a-sphere position="0 1.25 -5" radius="1.25" color="#EF2D5E" shadow></a-sphere>
      <a-cylinder position="1 0.75 -3" radius="0.5" height="1.5" color="#FFC65D" shadow></a-cylinder>
      <a-plane position="0 0 -4" rotation="-90 0 0" width="4" height="4" color="#7BC8A4" shadow></a-plane>
      <a-entity id="ctlL" laser-controls="hand: left"></a-entity>
      <a-entity id="ctlR" laser-controls="hand: right"></a-entity>
      <a-entity camera look-controls position="0 1.6 0">
        <a-text id="txt" value="" position="0 0 -1" scale="0.4 0.4 0.4" align="center" color="#000000"></a-text>
        <a-text id="txt2" value="" position="0 -0.15 -1" scale="0.4 0.4 0.4" align="center" color="#000000"></a-text>
      </a-entity>
    </a-scene>

    <script>
      const ctlL = document.getElementById("ctlL");
      const ctlR = document.getElementById("ctlR");
      const txt = document.getElementById("txt");
      const txt2 = document.getElementById("txt2");

      //Getting Position of Right Controller.
      const timer = setInterval(() => {     
        var p=ctlR.object3D.position;
        txt2.setAttribute("value","R-Position: "+ p.x.toFixed(2)+", "+p.y.toFixed(2)+", "+p.z.toFixed(2));
      }, 100);

      //Stick Moved
      ctlL.addEventListener('axismove',function(event){
        txt.setAttribute("value", "L Stick  x:"+event.detail.axis[0].toFixed(2)+", y:"+event.detail.axis[1].toFixed(2));
      });
      ctlR.addEventListener('axismove',function(event){
        txt.setAttribute("value", "R Stick  x:"+event.detail.axis[0].toFixed(2)+", y:"+event.detail.axis[1].toFixed(2));
      });

      //Trigger Touch Started
      ctlL.addEventListener('triggertouchstart', function (event) {
        txt.setAttribute("value","Left touch started ");
      });
      ctlR.addEventListener('triggertouchstart', function (event) {
        txt.setAttribute("value","Right touch started");
      });

      //Trigger Touch Ended
      ctlL.addEventListener('triggertouchend', function (event) {
        txt.setAttribute("value","Left touch ended ");
      });
      ctlR.addEventListener('triggertouchend', function (event) {
        txt.setAttribute("value","Right touch ended");
      });

      //Trigger Pressed
      ctlL.addEventListener('triggerdown', function (event) {
        txt.setAttribute("value","Left trigger down");
      });

      ctlR.addEventListener('triggerdown', function (event) {
        txt.setAttribute("value","Right trigger down");
      });

      //Trigger Released
      ctlL.addEventListener('triggerup', function (event) {
        txt.setAttribute("value","Left trigger up");
      });
      ctlR.addEventListener('triggerup', function (event) {
        txt.setAttribute("value","Right trigger up");
      });

      //Grip Pressed
      ctlL.addEventListener('gripdown', function (event) {
        txt.setAttribute("value","Left gripdown down");
      });
      ctlR.addEventListener('gripdown', function (event) {
        txt.setAttribute("value","Right gripdown down");
      });

      //Grip Released
      ctlL.addEventListener('gripup', function (event) {
        txt.setAttribute("value","Left gripdown up");
      });
      ctlR.addEventListener('gripup', function (event) {
        txt.setAttribute("value","Right gripdown up");
      });


      //A-buttorn Pressed 
      ctlR.addEventListener('abuttondown', function (event) {
        txt.setAttribute("value","Right A-button down");
      });

      //A-buttorn Released 
      ctlR.addEventListener('abuttonup', function (event) {
        txt.setAttribute("value","Right A-button up");
      });

      //B-buttorn Pressed 
      ctlR.addEventListener('bbuttondown', function (event) {
        txt.setAttribute("value","Right B-button down");
      });

      //B-buttorn Released 
      ctlR.addEventListener('bbuttonup', function (event) {
        txt.setAttribute("value","Right B-button up");
      });

      //Y-buttorn Pressed 
      ctlL.addEventListener('ybuttondown', function (event) {
        txt.setAttribute("value","Left Y-button down");
      });

      //Y-buttorn Released 
      ctlL.addEventListener('ybuttonup', function (event) {
        txt.setAttribute("value","Left Y-button up");
      });

      //X-buttorn Pressed 
      ctlL.addEventListener('xbuttondown', function (event) {
        txt.setAttribute("value","Left X-button down");
      });

      //X-buttorn Released 
      ctlL.addEventListener('xbuttonup', function (event) {
        txt.setAttribute("value","Left X-button up");
      });
    </script>
  </body>
</html>

sample2

コントローラにinput-listenerコンポーネントを追加。

<!DOCTYPE html>
<html>
  <!--https://github.com/TakashiYoshinaga/Oculus-Quest-Input-Sample-->
  <head>
    <meta charset="utf-8">
    <title>Oculus Quest Input</title>
    <meta name="description" content="Oculus Quest Input">
    <script src="https://aframe.io/releases/0.9.2/aframe.min.js"></script>
  </head>
  <body>
    <script>  
      AFRAME.registerComponent('input-listener', {
        //Definition of right or left hand as a controller's property.
        schema: { 
            hand: { type: "string", default: "" }
        },
        //Initialization
        init:function () {
          this.el.txt = document.getElementById("txt");
          this.el.txt2 = document.getElementById("txt2");
          //Stick Moved
          this.el.addEventListener('axismove',function(event){
            this.txt.setAttribute("value", "Stick  x:"+event.detail.axis[0].toFixed(2)+", y:"+event.detail.axis[1].toFixed(2));
          }); 
          //Trigger Touch Started
          this.el.addEventListener('triggertouchstart', function (event) {
            this.txt.setAttribute("value","Trigger touch started ");
          });
          //Trigger Touch Ended
          this.el.addEventListener('triggertouchend', function (event) {
            this.txt.setAttribute("value","Trigger touch ended");
          });
          //Trigger Pressed
          this.el.addEventListener('triggerdown', function (event) {
            this.txt.setAttribute("value","Trigger down");
          });
          //Trigger Released
          this.el.addEventListener('triggerup', function (event) {
            this.txt.setAttribute("value","Trigger up");
          });
          //Grip Pressed
          this.el.addEventListener('gripdown', function (event) {
            this.txt.setAttribute("value","Gipdown down");
          }); 
          //Grip Released
          this.el.addEventListener('gripup', function (event) {
            this.txt.setAttribute("value","Gripdown up");
          });
          //A-buttorn Pressed 
          this.el.addEventListener('abuttondown', function (event) {
            this.txt.setAttribute("value","A-button down");
          });
          //A-buttorn Released
          this.el.addEventListener('abuttonup', function (event) {
            this.txt.setAttribute("value","A-button up");
          });
          //B-buttorn Pressed
          this.el.addEventListener('bbuttondown', function (event) {
            this.txt.setAttribute("value","B-button down");
          });
          //B-buttorn Released
          this.el.addEventListener('bbuttonup', function (event) {
            this.txt.setAttribute("value","B-button up");
          });
          //Y-buttorn Pressed 
          this.el.addEventListener('ybuttondown', function (event) {
            this.txt.setAttribute("value","Y-button down");
          });
          //Y-buttorn Released
          this.el.addEventListener('ybuttonup', function (event) {
            this.txt.setAttribute("value","Y-button up");
          });
          //X-buttorn Pressed
          this.el.addEventListener('xbuttondown', function (event) {
            this.txt.setAttribute("value","X-button down");
          });
          //X-buttorn Released
          this.el.addEventListener('xbuttonup', function (event) {
            this.txt.setAttribute("value","X-button up");
          });
        },
        //called evry frame
        tick: function () {
          //Position of righ-hand controller is shown in real-time.
          if(this.data.hand=="right"){
            var p=this.el.object3D.position;
            this.el.txt2.setAttribute("value","R-Position: "+ p.x.toFixed(2)+", "+p.y.toFixed(2)+", "+p.z.toFixed(2));
          }
        }
    });

</script>

    <a-scene background="color: #FAFAFA">
      <a-box position="-1 0.5 -3" rotation="0 45 0" color="#4CC3D9" shadow></a-box>
      <a-sphere position="0 1.25 -5" radius="1.25" color="#EF2D5E" shadow></a-sphere>
      <a-cylinder position="1 0.75 -3" radius="0.5" height="1.5" color="#FFC65D" shadow></a-cylinder>
      <a-plane position="0 0 -4" rotation="-90 0 0" width="4" height="4" color="#7BC8A4" shadow></a-plane>
      <a-entity id="ctlL" laser-controls="hand: left" input-listener="hand: left"></a-entity>
      <a-entity id="ctlR" laser-controls="hand: right" input-listener="hand: right"></a-entity>
      <a-entity camera look-controls position="0 1.6 0">
        <a-text id="txt" value="" position="0 0 -1" scale="0.4 0.4 0.4" align="center" color="#000000"></a-text>
        <a-text id="txt2" value="" position="0 -0.15 -1" scale="0.4 0.4 0.4" align="center" color="#000000"></a-text>
      </a-entity>
    </a-scene>   
  </body>
</html>

3. CGとのインタラクション

コントローラを用いてテレポートやマニピュレーション、弾の発射、物理演算の適用などVRに必要そうな要素を盛り込んだサンプルをGitHubやGlitchで公開しています。ご興味があれば是非こちらもご活用ください。

Qiita
https://qiita.com/Tks_Yoshinaga/items/bd7f85a2eada4f6a9113

GitHub
https://github.com/TakashiYoshinaga/Oculus-Quest-Interaction-Sample

体験
https://quest-demo.glitch.me

Videoデモ

18
13
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
18
13