この記事はRICOH THETA Advent Calendar 2015 21日目の記事です。
MESHを使ってThetaのシャッターをリモートで切ることができるソフトウェアタグを作ります。
なお、THETA API v2 を利用するため現在対応している機種はTheta Sのみとなります。
##MESHって?
これ https://meshprj.com/
BluetoothでiOSデバイスとペアリングして使えるワイヤレス電子ブロック的なものです。
Move(動き)タグ、Button(ボタン)タグ等のハードウェアタグとソフトウェアタグを組み合わせたレシピを定義することができます。
先日SDK(ベータ版)が公開され、jsベースでのソフトウェアタグの開発が可能となりました。
これにより、オンライン開発環境でコードを書いて、自分の端末にダウンロードして使うことができるようになります。
また、iPhone版のアプリも提供されたので、iPadがなくても使えるようになり、より手軽に遊べるようになっています。
ソフトウェアタグの開発手順等は以下のリンクなどが参考になりました。
https://meshprj.com/sdk/doc/ja/ (要MESH Developerアカウント)
http://qiita.com/masato/items/159a35c5ce741b6f1058
##Theta Sのシャッターを切るソフトウェアタグ
上記のMESH SDKと開発環境を利用して、MESHのレシピ上からTheta Sのシャッターを切れるソフトウェアタグを書いてみました。
ボタンタグ等と組み合わせて、リモートシャッターが切れるようになります。
ソフトウェアタグの内容は以下のようになります(initialize、executeのみ記載)。
このほか入力をトリガーにするため、input connectorを一つ以上定義します。
return {
runtimeValues : { sessionId: "" }
}
//Theta Endpoint URL の設定(デフォルトは192.168.1.1)
var endPointURL = "http://192.168.1.1"
var startSessionJSONData = {
name: "camera.startSession"
}
ajax( {
url : endPointURL + "/osc/commands/execute",
type : "post",
data : JSON.stringify( startSessionJSONData),
contentType: "application/json",
dataType: "json",
timeout : 5000,
success : function (contents) {
//取得したセッションIDをruntimeValuesに保管
runtimeValues.sessionId = contents.results.sessionId;
log("Theta session started: " + runtimeValues.sessionId);
//CaptureModeのチェック
checkCaptureMode(endPointURL);
//シャッターを切る
takeThetaPicture(endPointURL);
//セッションのクローズ
closeThetaSession(endPointURL);
callbackSuccess( {
resultType : "continue",
runtimeValues : runtimeValues
});
},
error : function (XMLHttpRequest, textStatus, errorThrown) {
log(XMLHttpRequest + " | " + textStatus + " | " + errorThrown);
callbackSuccess({
resultType: "continue"
});
}
});
function checkCaptureMode(endPointURL) {
var getOptionsJSONData = {
name: "camera.getOptions",
parameters: {
sessionId: runtimeValues.sessionId,
optionNames: ["captureMode"]
}
}
ajax( {
url : endPointURL + "/osc/commands/execute",
type : "post",
data : JSON.stringify( getOptionsJSONData),
contentType: "application/json",
dataType: "json",
timeout : 5000,
success : function (contents) {
log("CaptureMode: " + JSON.stringify(contents.results.options.captureMode));
if(contents.results.options.captureMode != "image"){
log("Please change capture mode to take photo");
callbackSuccess({
resultType: "stop"
});
};
callbackSuccess( {
resultType : "continue"
});
},
error : function (XMLHttpRequest, textStatus, errorThrown) {
log("Error:" + textStatus + " " + errorThrown);
callbackSuccess({
resultType: "continue"
});
}
} );
}
function takeThetaPicture(endPointURL) {
var takePictureJSONData = {
name: "camera.takePicture",
parameters: {
sessionId: runtimeValues.sessionId
}
}
ajax( {
url : endPointURL + "/osc/commands/execute",
type : "post",
data : JSON.stringify( takePictureJSONData),
contentType: "application/json",
dataType: "json",
timeout : 5000,
success : function (contents) {
log("Took picture");
},
error : function (XMLHttpRequest, textStatus, errorThrown) {
log("Error:" + textStatus + " " + errorThrown);
callbackSuccess({
resultType: "continue"
});
}
} );
}
function closeThetaSession(endPointURL) {
var closeSessionJSONData = {
name: "camera.closeSession",
parameters: {
sessionId: runtimeValues.sessionId
}
}
ajax( {
url : endPointURL + "/osc/commands/execute",
type : "post",
data : JSON.stringify( closeSessionJSONData),
contentType: "application/json",
dataType: "json",
timeout : 5000,
success : function (contents) {
log("Theta session closed: " + runtimeValues.sessionId);
},
error : function (XMLHttpRequest, textStatus, errorThrown) {
log("Error:" + textStatus + " " + errorThrown);
callbackSuccess({
resultType: "continue"
});
}
} );
}
return {
resultType : "continue"
};
ちなみにMESHのexport機能で取り出したものは下記。
これを直接importしても使えるはずです。
{"formatVersion":"1.0","tagData":{"name":"ControlThetaS","icon":"","description":"This software tag enables mesh to control Ricoh Theta S camera, such as taking pictures ","functions":[{"id":"function_0","name":"TakePicture","connector":{"inputs":[],"outputs":[]},"properties":[],"extension":{"initialize":"return {\n\truntimeValues : { sessionId: \"\" }\n}","receive":"","execute":"//Theta Endpoint URL の設定(デフォルトは192.168.1.1)\nvar endPointURL = \"http://192.168.1.1\"\n\nvar startSessionJSONData = {\n\tname: \"camera.startSession\"\n} \n\najax( {\n\turl : endPointURL + \"/osc/commands/execute\",\n\ttype : \"post\",\n\tdata : JSON.stringify( startSessionJSONData),\n\tcontentType: \"application/json\",\n\tdataType: \"json\",\n\ttimeout : 5000,\n\tsuccess : function (contents) {\n\t\truntimeValues.sessionId = contents.results.sessionId;\n\t\tlog(\"Theta session started: \" + runtimeValues.sessionId);\n\t\tcheckCaptureMode(endPointURL);\n\t\ttakeThetaPicture(endPointURL);\n\t\tcloseThetaSession(endPointURL);\n\t\tcallbackSuccess( {\n\t\t\t\t\tresultType : \"continue\", \n\t\t\t\t\truntimeValues : runtimeValues\n\t\t});\n\t},\n\terror : function (XMLHttpRequest, textStatus, errorThrown) {\n\t\tlog(XMLHttpRequest + \" | \" + textStatus + \" | \" + errorThrown);\t\t\n\t\tcallbackSuccess({\n\t\t\tresultType: \"continue\"\n\t\t});\n\t}\n});\n\n//Theta sの撮影モードがimageモードになっていることを確認する\nfunction checkCaptureMode(endPointURL) {\n\tvar getOptionsJSONData = {\n\t\tname: \"camera.getOptions\",\n\t\tparameters: {\n\t\t\tsessionId: runtimeValues.sessionId,\n\t\t\toptionNames: [\"captureMode\"]\n\t\t}\n\t} \n\t\n\tajax( {\n\t\turl : endPointURL + \"/osc/commands/execute\",\n\t\ttype : \"post\",\n\t\tdata : JSON.stringify( getOptionsJSONData),\n\t\tcontentType: \"application/json\",\n\t\tdataType: \"json\",\n\t\ttimeout : 5000,\n\t\tsuccess : function (contents) {\n\t\t\tlog(\"CaptureMode: \" + JSON.stringify(contents.results.options.captureMode));\n\t\t\tif(contents.results.options.captureMode != \"image\"){\n\t\t\t\tlog(\"Please change capture mode to take photo\");\n\t\t\t\tcallbackSuccess({\n\t\t\t\t\tresultType: \"stop\"\n\t\t\t\t});\n\t\t\t};\t\n\t\t\tcallbackSuccess( {\n\t\t\t\t\tresultType : \"continue\"\n\t\t\t});\n\t\t},\n\t\terror : function (XMLHttpRequest, textStatus, errorThrown) {\n\t\t\tlog(\"Error:\" + textStatus + \" \" + errorThrown);\t\t\n\t\t\tcallbackSuccess({\n\t\t\t\tresultType: \"continue\"\n\t\t\t});\n\t\t}\n\t} );\n}\n\n//Theta のシャッターを切る\nfunction takeThetaPicture(endPointURL) {\n\tvar takePictureJSONData = {\n\t\tname: \"camera.takePicture\",\n\t\tparameters: {\n\t\t\tsessionId: runtimeValues.sessionId\n\t\t}\n\t} \n\n\tajax( {\n\t\turl : endPointURL + \"/osc/commands/execute\",\n\t\ttype : \"post\",\n\t\tdata : JSON.stringify( takePictureJSONData),\n\t\tcontentType: \"application/json\",\n\t\tdataType: \"json\",\n\t\ttimeout : 5000,\n\t\tsuccess : function (contents) {\n\t\t\t\tlog(\"Took picture\");\n\t\t},\n\t\terror : function (XMLHttpRequest, textStatus, errorThrown) {\n\t\t\tlog(\"Error:\" + textStatus + \" \" + errorThrown);\t\t\n\t\t\tcallbackSuccess({\n\t\t\t\tresultType: \"continue\"\n\t\t\t});\n\t\t}\n\t} );\n}\n\n//セッションをクローズする\nfunction closeThetaSession(endPointURL) {\n\tvar closeSessionJSONData = {\n\t\tname: \"camera.closeSession\",\n\t\tparameters: {\n\t\t\tsessionId: runtimeValues.sessionId\n\t\t}\n\t} \n\n\tajax( {\n\t\turl : endPointURL + \"/osc/commands/execute\",\n\t\ttype : \"post\",\n\t\tdata : JSON.stringify( closeSessionJSONData),\n\t\tcontentType: \"application/json\",\n\t\tdataType: \"json\",\n\t\ttimeout : 5000,\n\t\tsuccess : function (contents) {\n\t\t\tlog(\"Theta session closed: \" + runtimeValues.sessionId);\n\t\t\t},\n\t\terror : function (XMLHttpRequest, textStatus, errorThrown) {\n\t\t\tlog(\"Error:\" + textStatus + \" \" + errorThrown);\t\t\n\t\t\tcallbackSuccess({\n\t\t\t\tresultType: \"continue\"\n\t\t\t});\n\t\t}\n\t} );\n}\n\nreturn {\n\tresultType : \"continue\"\n};","result":""}}]}}
##利用例
ボタンタグと組み合わせたレシピ例。この状態でボタンを押すとThetaのシャッターが切られます。
##追加でやりたいことなど
-
以下のAdvent Calendarの記事でシャッター音を消す方法が紹介されていたので、ソフトウェアタグでできるようにしたい http://hima-zinn.hatenablog.com/entry/2015/12/13/000000
-
動画撮影用ソフトウェアタグも作りたい
##まとめ
THETA API v2はREST APIで触れるので敷居が低く、とっつきやすかったです。
なお、本題とはずれますがAPIの動作チェックはElasticsearch用のChromeプラグインSenseを利用するととても便利でした。