OpenCVでボールの輪郭描画(複数)

前回やった白いボールの輪郭描画に引き続き、赤いボールも同時に輪郭を描画してみます。

環境・バージョン

Windows10
Python 3.6.1
OpenCV 3.3.1

輪郭描画のコード

前回と処理の流れは変わらないのですが、複数回の処理を関数にまとめるなど、コードは若干整理しています。

ただ私自身、Pythonの構文も学びながら書くレベルなので参考程度に・・・

実行してみると・・・

実際は動画で、しっかりと転がる赤いボールを追尾して認識しています。

赤いボールは途中から転がってくるので、それまで白いボールの赤文字のロゴが誤認されてしまいました。

そこで、半径が75以上のものしかボールと認識しない条件を加えています。

赤を識別する問題

他にも少し問題となったのが、赤いボールの識別です。

H(色相)の範囲指定が、前回のようにはいきません。

赤のHは分断されているからです。

赤を指定するときは、0~10 かつ 170 ~ 180 というような条件式が必要です。

上記コードでは、getMask([最小H,最小S,最小V], [最大H,最大S,最大V])という関数を作り、最小Hに負の値を指定すると、赤の条件式に切り替わります。

赤の条件式は「pythonで赤い物体を認識しよう」を参考とさせて頂きました。

同じ色のボールが複数あったときは?

赤と白、一つずつのボール認識は、比較的簡単だと思います。

これが赤6つとかになると、ボール同士がくっついて輪郭が一体化し、誤認しそうです。

くっついたボール、陰になって一部見えないボールなど、イレギュラーな認識については、haar-like機械学習と組み合わせで、地道に認識精度を上げていくのでしょう。

 

地道なことはせず、次回はボールの軌道描画を行います。

OpenCVでボールの輪郭描画

前回の色抽出に引き続き、OpenCVを使って輪郭描画を行います。

動画の中からボールを認識して、円で囲います。

環境・バージョン

Windows10
Python 3.6.1
OpenCV 3.3.1

輪郭描画の流れ

独学&初心者で正しいか分かりませんが・・・輪郭描画の流れを書きます。
コードは最後に掲載。

前回、白色の抽出をした画像を元に加工します。

白と黒の2色にしてしまいます。

このままでは、余計なノイズも検出されてしまいます。

対策として、一番大きい領域だけ検出するようにします。

そして、検出した領域の輪郭に接する円を描画して、元画像と合成。

うまく、囲ってくれました。

輪郭描画コード

import cv2
import numpy as np

cap = cv2.VideoCapture('xxxxx.mp4') # 任意の動画
while(1):
    _, frame = cap.read()
    # HSVに変換
    hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)

    # 白のHSV範囲
    lower_white = np.array([0,0,100])
    upper_white = np.array([180,45,255])

    # 白以外にマスク
    mask_white = cv2.inRange(hsv, lower_white, upper_white)
    res_white = cv2.bitwise_and(frame,frame, mask= mask_white)

    # 輪郭抽出
    gray = cv2.cvtColor(res_white, cv2.COLOR_RGB2GRAY)
    ret, thresh = cv2.threshold(gray, 45, 255, cv2.THRESH_BINARY)
    imgEdge, contours, hierarchy = cv2.findContours(thresh, cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)

    # 一番大きい輪郭を抽出
    contours.sort(key=cv2.contourArea, reverse=True)
    cnt = contours[0]

    # 最小外接円を描く
    (x,y), radius = cv2.minEnclosingCircle(cnt)
    center = (int(x),int(y))
    radius = int(radius)
    img = cv2.circle(frame,center,radius,(0,255,0),2)

    # 再生
    cv2.imshow('video',img)
    k = cv2.waitKey(25) & 0xFF

    #Q で終了
    if k == ord('q'):
        break

cv2.destroyAllWindows()

今回も「OpenCV-Python チュートリアル文書」を参考とさせて頂きました。

ありがとうございました。

OpenCVで色抽出(白)

今回は久しぶりにOpenCVを使い、色の抽出を行います。

環境・バージョン

Windows10
Python 3.6.1
OpenCV 3.3.1

素材画像の用意

白、赤、青のボールを使った頭脳戦「ボッチャ」の動画を素材とします。
Webカメラでリアルタイムでも問題ありません。

コートが緑だと、色抽出が楽です。

白を抽出してみよう

赤、青の方が簡単そうですが、いきなり白の抽出を行います。

白は色相がないので、色抽出に不向きなような気がしますが・・・

先に結果を。

十分です!

HSVに変換して評価

色の抽出には、まず色情報を分かりやすいように、色相(H)、彩度(S)、明度(V)に分けて評価できるよう、HSV空間に変換します。

HSVはデザイナー寄りの人であれば、理解しやすい分野かと思います。

白の場合、色相は関係ないので0度~360度、すべてを対象としましょう。
※OpenCV3だと、色相を0~180で指定するみたいです。

彩度と明度は、0~255で指定。
白だと、彩度は低めで45以下。明度は影も考慮すると255~100くらい?

・・・試しながら調整したほうが早いです。

白抽出コード

import cv2
import numpy as np

cap = cv2.VideoCapture('xxxxx.mp4') # 任意の動画
while(1):
    _, frame = cap.read()
    # HSVに変換
    hsv = cv2.cvtColor(frame, cv2.COLOR_BGR2HSV)

    # 白のHSV範囲
    lower_white = np.array([0,0,100])
    upper_white = np.array([180,45,255])

    # 白以外にマスク
    mask_white = cv2.inRange(hsv, lower_white, upper_white)
    res_white = cv2.bitwise_and(frame,frame, mask= mask_white)

    # 再生
    cv2.imshow('res',res_white)
    k = cv2.waitKey(25) & 0xFF # 数字が小さいほど再生が速くなる

    #Q で終了
    if k == ord('q'):
        break

cv2.destroyAllWindows()

これで、動画から白を抽出できます。

HSVの値を変えれば、青でも赤でも抽出可能です。

 

意外と色抽出は簡単なので、クロマキー合成の自作とかも、気軽に楽しめそうです。

引き続き次回も同じ動画素材を使い、OpenCVで輪郭抽出します。

three.jsでボール軌道描画

Webで3Dを簡単に扱えるようになる「three.js」を使い、ボール軌道の描画に挑戦してみます。

なお私は、ボールの運動方程式をまったく知りません。

運動方程式を調べよう

ボールの運動方程式は、ネットで簡単に見つけられます。

とりあえず、検索1番目に出てきた「いろいろな運動」というPDFを開いてみました。

 

・・・よし、見なかったことにしよう。

イージングアニメーションを活用

今持っているWeb知識のみでどうにかします。

ボールの動きは、jQueryアニメーションのイージング「Easing function」を利用できるかもしれません。

バウンドは「easeOutBounce」、減速は「easeOutCubic」を使い、適当な<span>にアニメーションを割り当ててみます。

<span id="ballX"><span id="ballY" class="ball"></span><span id="ballZ" class="ball"></span></span>
var $ballX = $("#ballX"),
    $ballY = $("#ballY"),
    $ballZ = $("#ballZ");

$ballX.animate({
 'left': '500px'
},{
 'duration': 3000,
 'easing': 'easeOutCubic'
});
$ballY.animate({
 'top': '-200px'
},{
 'duration': 3000,
 'easing': 'easeOutCubic'
});
$ballZ.animate({
 'top': '100px'
},{
 'duration': 2000,
 'easing': 'easeOutBounce'
});

コートを上から見て、ボールの横位置がx、縦位置がy、高さがzとなる想定です。

あとは60ミリ秒おきに各spanの座標を配列に変換するスクリプトを実行すると・・・

[{"x":-368.33,"y":206.92,"z":107.49},{"x":-336.53,"y":194.17,"z":105.6},{"x":-306.63,"y":182.19,"z":102.08},{"x":-284.88,"y":173.47,"z":98.32},{"x":-256.94,"y":162.27,"z":91.8},{"x":-236.98,"y":154.27,"z":85.82},{"x":-211,"y":143.86,"z":76.17},{"x":-187.42,"y":134.39,"z":65.33},{"x":-170.08,"y":127.45,"z":55.89},{"x":-147.94,"y":118.56,"z":41.77},{"x":-132.19,"y":112.25,"z":30.13},{"x":-112.11,"y":104.2,"z":13.03},{"x":-93.44,"y":96.72,"z":13.82},{"x":-80.02,"y":91.33,"z":19.33},{"x":-62.98,"y":84.5,"z":25.25},{"x":-50.95,"y":79.67,"z":28.57},{"x":-35.72,"y":73.56,"z":31.5},{"x":-21.69,"y":67.92,"z":32.75},{"x":-11.69,"y":63.92,"z":32.6},{"x":0.91,"y":58.86,"z":30.91},{"x":9.72,"y":55.33,"z":28.55},{"x":20.77,"y":50.89,"z":23.89},{"x":30.83,"y":46.86,"z":17.66},{"x":37.92,"y":44,"z":11.83},{"x":46.73,"y":40.45,"z":10.07},{"x":52.83,"y":38.02,"z":12.33},{"x":60.36,"y":34.98,"z":13.91},{"x":67.11,"y":32.28,"z":13.78},{"x":71.88,"y":30.36,"z":12.57},{"x":77.52,"y":28.09,"z":9.53},{"x":81.39,"y":26.55,"z":8.49},{"x":86.08,"y":24.66,"z":9.36},{"x":90.17,"y":23.02,"z":8.58},{"x":92.94,"y":21.89,"z":7.8},{"x":96.22,"y":20.58,"z":7.8},{"x":98.38,"y":19.7,"z":7.8},{"x":100.89,"y":18.69,"z":7.8},{"x":103,"y":17.84,"z":7.8},{"x":104.34,"y":17.3,"z":7.8},{"x":105.88,"y":16.69,"z":7.8},{"x":106.52,"y":16.42,"z":7.8},{"x":107.81,"y":15.89,"z":7.8},{"x":108.59,"y":15.58,"z":7.8},{"x":109.03,"y":15.41,"z":7.8},{"x":109.47,"y":15.22,"z":7.8},{"x":109.69,"y":15.14,"z":7.8},{"x":109.86,"y":15.06,"z":7.8},{"x":109.95,"y":15.02,"z":7.8},{"x":109.98,"y":15.02,"z":7.8},{"x":110,"y":15,"z":7.8},{"x":110,"y":15,"z":7.8}]

三次元の座標配列が書き出されました。

three.jsでボール軌道を描画

座標データさえできれば、ボールの軌道を描画していくのは簡単です。

60ミリ秒おきに、半透明のボールを描画していきます。

window.addEventListener('DOMContentLoaded', init);

function init() {
  "use strict";
  var canvasID = "#court",
      width = 1280,
      height = 720,
      ballSize = 8.6, //ボール直径
      ballDents = 0.8, //ボールのへこみ
      balls = {};
  
  var orbit = [{"x":-368.33,"y":206.92,"z":107.49},{"x":-336.53,"y":194.17,"z":105.6},{"x":-306.63,"y":182.19,"z":102.08},{"x":-284.88,"y":173.47,"z":98.32},{"x":-256.94,"y":162.27,"z":91.8},{"x":-236.98,"y":154.27,"z":85.82},{"x":-211,"y":143.86,"z":76.17},{"x":-187.42,"y":134.39,"z":65.33},{"x":-170.08,"y":127.45,"z":55.89},{"x":-147.94,"y":118.56,"z":41.77},{"x":-132.19,"y":112.25,"z":30.13},{"x":-112.11,"y":104.2,"z":13.03},{"x":-93.44,"y":96.72,"z":13.82},{"x":-80.02,"y":91.33,"z":19.33},{"x":-62.98,"y":84.5,"z":25.25},{"x":-50.95,"y":79.67,"z":28.57},{"x":-35.72,"y":73.56,"z":31.5},{"x":-21.69,"y":67.92,"z":32.75},{"x":-11.69,"y":63.92,"z":32.6},{"x":0.91,"y":58.86,"z":30.91},{"x":9.72,"y":55.33,"z":28.55},{"x":20.77,"y":50.89,"z":23.89},{"x":30.83,"y":46.86,"z":17.66},{"x":37.92,"y":44,"z":11.83},{"x":46.73,"y":40.45,"z":10.07},{"x":52.83,"y":38.02,"z":12.33},{"x":60.36,"y":34.98,"z":13.91},{"x":67.11,"y":32.28,"z":13.78},{"x":71.88,"y":30.36,"z":12.57},{"x":77.52,"y":28.09,"z":9.53},{"x":81.39,"y":26.55,"z":8.49},{"x":86.08,"y":24.66,"z":9.36},{"x":90.17,"y":23.02,"z":8.58},{"x":92.94,"y":21.89,"z":7.8},{"x":96.22,"y":20.58,"z":7.8},{"x":98.38,"y":19.7,"z":7.8},{"x":100.89,"y":18.69,"z":7.8},{"x":103,"y":17.84,"z":7.8},{"x":104.34,"y":17.3,"z":7.8},{"x":105.88,"y":16.69,"z":7.8},{"x":106.52,"y":16.42,"z":7.8},{"x":107.81,"y":15.89,"z":7.8},{"x":108.59,"y":15.58,"z":7.8},{"x":109.03,"y":15.41,"z":7.8},{"x":109.47,"y":15.22,"z":7.8},{"x":109.69,"y":15.14,"z":7.8},{"x":109.86,"y":15.06,"z":7.8},{"x":109.95,"y":15.02,"z":7.8},{"x":109.98,"y":15.02,"z":7.8},{"x":110,"y":15,"z":7.8},{"x":110,"y":15,"z":7.8}];
  
  function createBall(x,y,z,color,id){
    var colorData;
    id = id || "";
    id = color + id;
    if(color === "red"){
      colorData = 0xb71c1c;  
    }
    else if(color === "blue"){
      colorData = 0x0D47A1;
    }
    else{
      colorData = 0xFFFFFF;
    }
    var geometry = new THREE.SphereGeometry(ballSize, 24, 24);
    var material = new THREE.MeshLambertMaterial({color: colorData});
    balls[id] = new THREE.Mesh(geometry, material);
    balls[id].position.set(x, y, z);
    scene.add(balls[id]);
  }
  
  function createOrbitBall(x,y,z,color){
    var colorData;
    if(color === "red"){
      colorData = 0xb71c1c;  
    }
    else if(color === "blue"){
      colorData = 0x0D47A1;
    }
    else{
      colorData = 0xFFFFFF;
    }
    var geometry = new THREE.SphereGeometry(ballSize, 12, 12);
    var material = new THREE.MeshLambertMaterial({
      color: colorData, wireframe: true, 
      transparent: true,
      opacity: 0.1}
    );
    var orbitBall = new THREE.Mesh(geometry, material);
    orbitBall.position.set(x, y, z);
    scene.add(orbitBall);
  }
  
  function createCourt(){
    var loader = new THREE.TextureLoader();
    loader.load("common/img/court.png", function(texture){
      readiness(texture);
    });
    function readiness(texture){
      var geometry = new THREE.PlaneGeometry(1258, 608);
      var material = new THREE.MeshPhongMaterial({color: 0xFFFFFF,map: texture});
      var ground = new THREE.Mesh(geometry, material);
      scene.add(ground);
      camera.lookAt(ground.position);
    }
  }

  // レンダラー作成
  var renderer = new THREE.WebGLRenderer({
    canvas: document.querySelector(canvasID),
    antialias: true
  });
  renderer.setPixelRatio(window.devicePixelRatio);
  renderer.setSize(width, height);

  // シーン作成
  var scene = new THREE.Scene();

  // カメラ作成
  var camera = new THREE.PerspectiveCamera(45, width / height, 1, 10000);
  camera.position.set(0, -700, 500);

  // ボール作成
  var z = ballSize - ballDents;
  createBall(-440, 215, z, "jack");
  createBall(-390, 215, z, "red", 1);
  createBall(-415, 215, z, "red", 2);
  createBall(-390, 20, z, "red", 3);
  createBall(-415, 20, z, "red", 4);
  createBall(-390, -175, z, "red", 5);
  createBall(-415, -175, z, "red", 6);
  createBall(-390, 115, z, "blue", 1);
  createBall(-415, 115, z, "blue", 2);
  createBall(-390, -80, z, "blue", 3);
  createBall(-415, -80, z, "blue", 4);
  createBall(-390, -275, z, "blue", 5);
  createBall(-415, -275, z, "blue", 6);
  createCourt();
  
  // 平行光源
  var light = new THREE.DirectionalLight(0xFFFFFF);
  light.intensity = 2;
  light.position.set(-2, -1, 10);
  scene.add(light);

  // 環境光源
  var ambient = new THREE.AmbientLight(0x222222);
  ambient.intensity = 4;
  scene.add(ambient);

  // アニメーション実行
  var i = 0;
  var animOrbit = function(){
    
    if(i < orbit.length){
      balls.jack.position.set(orbit[i].x, orbit[i].y, orbit[i].z);
      createOrbitBall(orbit[i].x, orbit[i].y, orbit[i].z,"jack");
      renderer.render(scene, camera);
      i++;
    }
    else{
      clearInterval(setOrbit);
    }
  };

  var setOrbit = window.setInterval(animOrbit, 60);
}

実行してみると・・・

アニメーションしながら、自然なボールの軌道を描画することができました。

さらにカメラをマウスで操作できるように改修し、いろいろなアングルで見られるようにしました。

最終的に、現実のコートで複数カメラで撮影した動画を、OpenCVで三次元座標に変換し、ボールの軌道を可視化します。

なので、ボールの軌道を自作する必要はなかったのですが・・・試しです。

Webで3D three.jsを学ぶ

枯れた技術が大好きです。

今回は2011年に話題のピーク(Googleトレンドで調査)だった、「WebGL」を使い、3Dに挑戦しようと思います。

three.jsという ありがたいライブラリ

WebGLを簡単に利用できる「three.js」というライブラリがあります。
これを使い、3DCGを制作します。

世界で長期的に愛用されていて、情報もたくさんあるので学習コストは低いでしょう。

3Dを使う理由

Web系の仕事をしている人にとって、3Dを使う場面はあまりないと思います。
(WebVRとか出てきて、そんなことないかも)

単純にWebサイトのUIデザインを3Dにすると、操作性が落ちます。
マウスにしろタップにしろ、入力操作はxyの2Dですし、認識としても、立体より平面の方が理解しやすいからです。

さらに開発工数も上がりますし、なかなか3Dの使いどころが難しいです。

 

私の場合、それでも3Dに手を出したのは、明確な目的があったからです。

パラリンピック正式種目に「ボッチャ」という球技があるのですが、3次元で繰り広げられる頭脳戦を可視化できれば・・・(以下略)

three.jsを使うための必要知識

球や立方体を配置して、カメラと光源を調整するくらいなら、そこまで知識はいりません。

  • HTML
  • JavaScript
  • 3Dモデリング

どれも初級以上のレベルでなんとかなるような気がします。

3Dモデリングは例え未経験でも、カメラ、光源を調整しながら、理解していけるものだと思います。

three.jsの利用準備

three.jsを利用するための準備は非常に簡単です。

任意の場所にHTMLを作成します。

HTML作成

<!DOCTYPE html>
<html lang="ja">
<head>
<meta charset="UTF-8">
<title>three.jsを学ぶ</title>
<script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/88/three.min.js"></script>
<script src="sample.js"></script>
</head>
<body>
 <canvas id="sample"></canvas>
</body>
</html>

6行目で(CDNから)three.jsを読み込むだけです。

7行目のsample.jsで書いたコードを、10行目のcanvasに描画する流れとなります。

sample.js作成

ボールを作ってみます。

window.addEventListener('DOMContentLoaded', init);

function init() {
  "use strict";
  var canvasID = "#sample",
      width = 1280,
      height = 720;
  
  // シーン作成
  var scene = new THREE.Scene();

  // ボール作成
  var geometry = new THREE.SphereGeometry(100, 24, 24);
  var material = new THREE.MeshLambertMaterial({color: 0xFFFFFF});
  var ball = new THREE.Mesh(geometry, material);
  scene.add(ball);

  // カメラ作成
  var camera = new THREE.PerspectiveCamera(45, width / height, 1, 10000);
  camera.position.set(0, 0, 1000);
  
  // 光源作成
  var light = new THREE.DirectionalLight(0xFFFFFF);
  light.position.set(-1, 1, 1);
  scene.add(light);
  
  // レンダリング(描画)
  var renderer = new THREE.WebGLRenderer({
    canvas: document.querySelector(canvasID),
    antialias: true
  });
  renderer.setPixelRatio(window.devicePixelRatio);
  renderer.setSize(width, height);
  renderer.render(scene, camera);
}

3D空間(シーン)を作って、ボールを置いて、カメラ向けて、照明つけて、レンダリングする流れです。

レンダリング結果

闇に浮かぶ球体ができました。

基本が分かれば、冒頭のボッチャコートも2時間くらいで出来上がります。

 

次回はボールの軌道の描画です。

MOVERIO「BT-300」と「BT-350」の耐久差

MOVERIO「BT-300」で加速度センサーとジャイロセンサーを使うWebアプリを作りました。

ただ、2~3時間使うと、センサーの値が取得できなくなってしまいます。

耐久テストロボットの制作

加速度やジャイロなどのモーションセンサーは、常に値を取得する必要があるので負荷がかかります。

あるイベントで長時間利用することになり、モーションセンサーを取得したWebアプリの耐久力を調査する必要がでてきました。

もちろん、MOVERIOをかけて、数時間首を振り続ける耐久力が私にはないので、Arduinoで制作したロボットを利用します。

「BT-300」は2~3時間でフリーズ

モーションセンサーだけが原因とは言えませんが、「BT-300」で動かしたWebアプリでは、2~3時間でまったく加速度やジャイロの値が取得できなくなってしまいました。

電源を落として、再起動をすると復活します。

「BT-300」は個人利用向けなので、そこまで耐久力がないのかもしれません。

「BT-350」は6時間以上稼働

比べて商用利用を想定した「BT-350」の方は、6時間以上動かしてもまったく問題ありません。

さすがの商用利用の耐久力です。

「BT-350」は「BT-300」の倍くらいの値段ですが、その分、見えない部分で相当改良されているのだと思います。

運用トラブル、開発トラブルを考慮すると、「BT-350」の方が安上がりかもしれません。

加速度センサーとジャイロセンサーを可視化

スマホの普及で、加速度センサーとジャイロセンサーが身近になりました。

このセンサーたちをうまく使いこなすため、3軸加速度センサー と3軸ジャイロセンサーを搭載した「MPU-6050」モジュールを使い、動作確認しながら理解を深めようと思います。

MPU-6050モジュールの接続

Arduino Nanoを利用します。A4、A5につなぐI2C接続です。

プログラムは「加速度+ジャイロのGY-521(MPU-6050)を使ってみた -1-」の掲載コードを利用させて頂きました。

シリアルモニタで確認すると・・・

うっ・・・大量の数字が高速で書き出されます。数字を追えません。

だいたい、1行に何で11個も値が書き出されるのでしょう?

加速度3軸+ジャイロ3軸、計6軸のはずです。

気温や角度も取得

コードを見ると、加速度とジャイロの他に、気温と角度が表示されるようです。(最初の0は無視します。)

どんなに振り回しても、1番最初の値が変わらないと思ったら気温でした。
気温を出力する理由がありそうですが、深追いはやめておきましょう。

2、3、4番目の値は加速度で、振り回すと値が激しく変化します。
センサーを静止しても重力の方向に、加速度が加わるようです。

次に5、6、7番目の角度の値がですが・・・これはなんでしょう。

acc(加速度)_angle(角度)なので、各軸の重力のかかり方から傾きの角度を算出できるだと思います。

そして最後に8、9、10番目がジャイロの値です。
回転速度が出力されます。

静止状態で0のはずですが、誤差があるようです。

センサーデータ可視化の全体構成

数字の羅列から、加速度やジャイロの変化を読み取るのは厳しいです。

そこで、センサーデータを可視化してみたいと思います。

MPU-6050で取得する加速度とジャイロのセンサーデータを、Node.js経由でブラウザに書き出します。

JavaScriptとCSSを使い、センサーの変化をアニメーションで表現します。

加速度は白いボール、角度はピンクのひし形、ジャイロは歯車で表現します。

まずは加速度の確認

ぐるんっと回りました。

横軸X、縦軸Y、ボールの遠近でZを表現しています。

数字の羅列では分かりませんでしたが、可視化することによって、正常な値を出力していたことが分かります。

次に角度の確認

角度を使い、傾きを再現してみます。

これは、少々扱いが難しいです。
繊細に扱わないと、思わぬ方向を向いてしまいます。
何か補正が必要か・・・?

使いこなせれば、3Dオブジェクトを操作できそうです。

最後にジャイロの確認

早く動かすほど、歯車が高速で回るようにしています。

ジャイロはだいたい意図通りの動きをします。

基礎を理解して応用へ

以上、大量のデータを可視化することで、加速度センサー、ジャイロセンサーの基礎が学べたと思います。

これらのセンサーデータを、IoTやアプリ開発に応用していきます。

MOVERIOで首の振り連動ロボット

前回から作っている恐竜ロボットの試験動作を兼ね、ちょっと遊びます。

スマートグラス「MOVERIO」をかけた子供の首の振りに合わせて、恐竜も首を振るようにします。

MOVERIOと連動で首振り

子供が上を向いたら、恐竜も上を向きます。

MOVERIOのセンサーと連動して、Arduinoで作った恐竜ロボットが動きます。

サーボモーターをNode.jsで制御

この、まぬけな顔の恐竜の頭部は、Arduino Nanoが組み込まれていて、サーボモーターを制御します。(中身は前回の記事に写真があります。)

ベロにはレーザー距離センサーが実装されていますが、今回は使いません。

後頭部はPCと接続するジャックがあり、USBで接続したPC(Node.jsサーバー)から、4方向(正面、右、左、上)向きの命令を受け取ります。

加速度センサーとジャイロセンサーで首の向きを

MOVERIOの加速度センサーで首の上下、ジャイロセンサーで首の振りを取得します。

MOVERIOのブラウザでHTMLを読み込んで、JavaScriptでセンサー値を取得し、Node.jsサーバーにSocket.ioで送っています。

MOVERIO(JavaScriptで首の方向取得)
↓無線LAN
PC(Node.jsでリアルタイムシリアル通信)
↓有線USB
恐竜ロボット(Arduinoでサーボモーター動作)

という流れです。

使いどころは?

今回の試作は、カメラ、ロボットの遠隔操作に使えそうです。
ローカルサーバーでの実験でしたが、リアルタイム性は高いです。遅延なしでMOVERIOとサーボモーターが連動していました。

遊びのように見えて、案外、必要性が高い分野かもしれません。

超高齢社会の日本ではロボットの補助が必須になるでしょう。

Webの知識でロボット制御は、近い将来、役に立ちそうな・・・そんな予感がします。

Arduinoをぬいぐるみに

無機質なArduinoを、フェルトのぬいぐるみでかわいくします。

ボッチャソナー2号の続きです。

恐竜のぬいぐるみ

Arduinoを覆う、恐竜のぬいぐるみを作ります。

口を開けて距離を測る・・・恐竜ソナーとなる予定です。

フェルトを切ります。

市販のぬいぐるみ用型紙をベースにしつつ、自身のArduino工作に合うよう、アレンジが必要です。

恐竜の頭です。

恐竜の頭に、Arduino Nanoを収納します。

ブレッドボードは大きいので、ユニバーサル基板を、頭のサイズにカットして使います。

距離センサーは恐竜の口からベロンと出したいと思います。

無駄に長い配線はカットします。後戻りできないので勇気がいります。

カットした配線を割いて・・・

基板にはんだ付けします。

結構ぐちゃぐちゃになってきました。

量産できるような採算性の良い開発であれば、プリント基板でシンプルにできるのですが・・・

ベロをつけます。

なぜか逆光の怖い写真ですが、目をつけるとかわいい(若干まぬけな)感じになります。

つづく・・・。

ラジコンの視界をスマートグラスで

子供の誕生日プレゼントで、ラジコン戦車を買いました。

スマートグラスを使い、ラジコン戦車に乗っているような視界を見ながら、操縦できるようにします。

撮影映像を見ながらの操縦

スマートグラスと「MOVERIO」は、アプリを使ってドローンの撮影映像を目の前で確認できるという、特長紹介がされています。

ドローンは持っていませんが、たまたま子供がラジコン戦車を誕生日に欲しがっていたので、これを使って擬似体験をしようと思います。

システム構成

前回の構成に、ラジコンが加わっただけです。

システム環境やコードなどは、前回の記事をご確認ください。

ラジコンにWebカメラ搭載

たまたまですが、形がはまって、簡単に装着できました。

Webカメラの有線が気に入らない場合、無線のWebカメラに置き換えたほうがよいでしょう。

 

あとは、前回と同様の手順でサーバーを立ち上げ、ノートパソコンとスマートグラスのブラウザで、カメラ映像を表示します。

くっ・・・楽しい

戦車を操縦しながら、戦車の視界を見ることができます。

この戦車、砲身から球を撃てるので、戦車の視界で狙いを定め、発砲することもできます。

的を用意して、ゲームを作れます。

※子供のスマートグラスの利用は、年齢制限と時間制限に十分ご注意ください。

作業向きのスマートグラス+AR

スマートグラス+AR(視界の拡張)を、何か操縦する系統のゲームで利用を考えると、臨場感がVRに劣り、いまいちでしょう。

スマートグラス+ARである良さは、操縦している対象と、その対象の視界を同時に見れること。

ゲームより、物販作業とか救出作業とかのほうが向いています。

 

スマートグラスを購入して、ラジコンで遊んで終わった・・・という結果にならないよう、需要ある開発を考えていきたいです。