ご覧いただきありがとうございます。
この記事はSB-AI Advent Calendar 2019の18日目の記事となります。
#はじめに
突然ですが皆様は「人工生命」と聞いてどんなものをイメージしますか?
SFに出てくる怪物のようなイメージを持つ方も多いかもしれませんが、家を掃除してくれるルンバも人工生命の一つと言えます。
「人工生命」は国際的には「ALIFE」という名称で世界各国で研究が進められており、人工的に生命を作る事を通じて生命とは何かを理解しようとする研究分野の事です。
私はこの分野の研究者でもなんでもない、ただのサラリーマンですが、仕事でロボットに関わる機会が多く、
また昨年ALIFE関連のハッカソンに参加しチームで作品を作った経験から、プライベートで「人工生命」を1つのテーマとしてモノづくりをしております。
(嬉しい事にチームで作った作品が、今年の文化庁メディア芸術祭のアート部門で審査委員会推薦作品として選出されました!動画はこちら)
今回の記事は、私が最近注目している「toio(トイオ)」というロボットのあるアプリに着目し、それを参考にtoioで生き物っぽい何かを作ってみた。という記事です。
※この記事は筆者がtoioで遊んでみて勝手に思った事を書いているだけの記事となりますので、ALIFEの研究記事等ではございません。ご了承ください。
#toioについて
toioはsonyから2019年3月に発売されたロボットトイです。
※画像はこちらより引用:
公式サイトを見ていただけると一目瞭然ですが、主に子供向けのプログラミングトイとしての用途が主で、
カセットを用いて遊べる内容を変えたり、白いキューブにレゴなどを用いてオリジナルな何かを作ったりできるようになっています。
またスクラッチベースのビジュアルプログラミング環境の提供もされています。
画像にある白いキューブが「toio core cube」と言われるロボット本体なのですが、このcubeの仕様はネット上に公開されており、
またtoio.jsというnode.js用のライブラリも合わせて提供されていたり等、子供から上級者な大人まで遊べるようになっています。
今後も新たなエンタメ系ソフトやアプリの登場が予定されており、今後の展開が楽しみなtoioですが、
個人的に気になったtoio用スマホアプリが11月から提供開始となりました。
##「ウロチョロス」現る
11月12日にtoioの新プロジェクトの発表があり、その際に発表された一つに「ウロチョロス」というアプリがありました。
(開発元のモリカトロンさんの記事より引用:https://morikatron.ai/2019/11/toio_urochoros/)
当時はtoioを買おうか迷っていたものの、TLに流れてきた下記のツイートの動画たちを見てすぐにamazonで購入しました。
生き物っぽいものを作りたい私にとって、まさに目指すべき一例でしたので即決でした。
ウロチョロスたちが鬼ごっこをしています。よく見ると個体ごとの性格の違いが見えるかも。#toio pic.twitter.com/6meUmeykBM
— モリカトロンAIラボ (@morika_ailabo) November 12, 2019
動画では鬼ごっこをしていますが、この他にも自由にcube達に動き回ってもらうモードや、ユーザーの表情判定をcube達が行うモードがあったりします。
(※顔の撮影はスマホで行う)
なお、ユーザーの表情判定は純粋に画像認識したときのスコアで順位をつけているのではなく、最終的に個々のcubeが持つ評価基準を用いて判定が下されるのだとか。
##ウロチョロスの生き物らしさとは
自分も実際にウロチョロスを試してみて、自由に動き回るモードでは様々なcube達のやり取りが観察できました。
・会話をしているようなアクション
・モノマネ(一方のcubeが他方のcubeに技を教える)
・パワー切れのcubeに対してのサポートアクション
・ダンス 等。
ウロチョロス観察日記その1 pic.twitter.com/N8JUm0kXnk
— MARVO@anima (@marvo737) December 22, 2019
ウロチョロス観察日記その2 pic.twitter.com/XnxS6CFTY3
— MARVO@anima (@marvo737) December 22, 2019
そこでわかってきたのは、ウロチョロスはcube同士の相互作用によって、生き物らしく見せているのではないか、という点でした。
コードは確認できないため、あくまでも想像ではありますが、一方のアクションが他方のアクションに影響を及ぼすような制御が多くなされていると見受けられます。
ここで考えたいのは、双方のアクションを把握するためには、toioがどこにいるかなど、周辺環境の情報取得の方法です。
昨今のロボットはカメラやセンサーによって周辺環境位を把握し自己位置推定等を行いますが、toioには見た目の通りカメラなどはありません。
そこでポイントになってくるのが「プレイマット」です。
##toioのプレイマットについて
※画像はこちらより引用
ウロチョロスで遊ぶには、「トイオコレクション」と言う別売のキットに付属する「プレイマット」と呼ばれる台紙が必要となります。
この台紙はただの台紙ではなく、紙面上に座標を示すPositionIDが印刷されている特殊な台紙となっています。
cubeにはそのPositionIDを読み出すセンサーが搭載されており、そのおかげでcubeの現在のx,y座標、さらにはcubeのx軸に対しての時計回りの角度を利用したプログラミングをすることが可能です。
つまり、このプレイマット上であればtoioを自由に操ることが可能であり、次からは2台のcubeの座標データを用いてそれっぽいものを作ってみたいと思います。
##生き物っぽい何かを作ってみる
■実行環境
・macOS Catalina 10.15.2
・Node.js v12.13.0
・toio.js 42ae66a
細かいセットアップ及び関数の説明については省略いたします。
下記が今回作成したコードとなります。toio.jsのサンプルをベースとしています。
const { NearScanner } = require('@toio/scanner')
// calculate chasing cube's motor speed
function chase(jerryX, jerryY, tomX, tomY, tomAngle) {
const diffX = jerryX - tomX
const diffY = jerryY - tomY
const distance = Math.sqrt(diffX * diffX + diffY * diffY)
if (distance < 70) {
return [0, 0] // stop
}
let relAngle = (Math.atan2(diffY, diffX) * 180) / Math.PI - tomAngle
relAngle = relAngle % 360
if (relAngle < -180) {
relAngle += 360
} else if (relAngle > 180) {
relAngle -= 360
}
const ratio = 1 - Math.abs(relAngle) / 90
let speed = 80
if (relAngle > 0) {
return [speed, speed * ratio]
} else {
return [speed * ratio, speed]
}
}
async function main() {
// start a scanner to find nearest two cubes
const cubes = await new NearScanner(2).start()
// connect two cubes (tom chases jerry)
const jerry = await cubes[0].connect()
const tom = await cubes[1].connect()
//set movable area
const min_x = 100
const max_x = 400
const min_y = 100
const max_y = 400
// set light color and store position
let jerryX = 0
let jerryY = 0
let jerryAngle = 0
jerry.turnOnLight({ durationMs: 0, red: 255, green: 0, blue: 255 })
jerry.on('id:position-id', data => {
jerryX = data.x
jerryY = data.y
jerryAngle = data.angle
})
// set light color and store position
let tomX = 0
let tomY = 0
let tomAngle = 0
tom.turnOnLight({ durationMs: 0, red: 0, green: 255, blue: 255 })
tom.on('id:position-id', data => {
tomX = data.x
tomY = data.y
tomAngle = data.angle
})
const randRange = (min, max) => Math.floor(Math.random() * (max - min) + min);
var random_x = 100
var random_y = 100
setInterval(() => {
random_x = randRange(min_x,max_x)
random_y = randRange(min_y,max_y)
},2000)
setInterval(() => {
jerry.move(...chase(random_x, random_y, jerryX, jerryY, jerryAngle), 100)
console.log("jerryX : " + random_x + " jerryY : " + random_y + " jerryAngle" + jerryAngle + "\n");
}, 50)
setInterval(() => {
tom.move(...chase(jerryX, jerryY, tomX, tomY, tomAngle), 100)
}, 100)
}
main()
これで動かしてみるとこのような動きになります。
自作ウロチョロスもどき pic.twitter.com/tczSdCawtk
— MARVO@anima (@marvo737) December 22, 2019
X,Y座標を指定範囲内でランダムに一定時間で変化させ、cube1(jerry)がその座標を追いかけて、cube2(tom)がcube1を追いかける、といった内容です。
ただ、この状態だとcube同士がよくぶつかるため、cube内のモーションセンサで衝突検知して距離を取るなどしてあげないといけないのですが、そちらの実装はまた次回にしたいと思います。。
それができてやっと相互作用について少し語れるようになるかと思うのですが、時間の都合上今回はここまでといたします。
#最後に
今回はウロチョロスアプリの影響を受け、toioで生き物っぽい(定義が曖昧で申し訳ありません、、)何かを作ってみました。
本当はもう少しALIFEと絡めることができれば良かったのですが、完全に力不足です。。修行あるのみですね。。
toioのcubeとプレイマットの組み合わせは、まだまだ面白そうなものが作れそうな予感がします。
マット上であれば座標だけでなく角度も取得できるので、cubeを出力デバイスではなく、入力デバイスとして使ってみるのも面白そうです。
(個人的にはTouchDesignerのコントローラーにしてみたいと考えています。)
以上、ありがとうございました。