できること
MESH*アプリ上で、処理をミリ秒単位で指定してsleepさせることが出来ます。
これを利用する事でミリ秒単位でずらしたアクションを実行するというニッチなニーズに答えることができます。
*MESHとは、小さな便利を形にできる、ブロック形状の電子タグです。
真中の砂時計アイコンが今回作ったカスタムタグです。青い枠が現在処理されているタグなのですが、作成したタグが意図通りLEDで設定している1.5秒と同じ時間だけ処理をsleepさせて、その次のアクションを実行しています。
新たに作った理由
①ロジックのタイマータグで待つ(sleep)では、秒単位の指定しかできない
②LEDタグの点灯する時間を利用すれば、0.1秒単位で指定できるがそのためだけにLEDタグを使うのも微妙
実装
色々と試した結果、Promiseを使って期待したsleepの動作が実現しました。
これによって、メイン処理を止めずに必要な所だけ待機指定ができます。
importファイル
{"formatVersion":"1.0","tagData":{"name":"wait","icon":"","description":"","functions":[{"id":"function_0","name":"ミリ秒単位で待つ時間を指定","connector":{"inputs":[{"label":""}],"outputs":[{"label":""}]},"properties":[{"name":"時間(ミリ秒)","referenceName":"millisecond","type":"number","defaultValue":"1000"}],"extension":{"initialize":"","receive":"","execute":"wait().then((v) => {\n log(\"wait \" + v + \" millisecond\");\t\n\t\n\tcallbackSuccess( {\n\t\tresultType : \"continue\"\n\t} );\n});\n\nfunction sleep() {\n\treturn new Promise(resolve => setTimeout(resolve,properties.millisecond));\n}\n\nasync function wait() {\n\t//非同期処理が完了するまで待機\n\tawait sleep();\n\t\n\treturn properties.millisecond.toString();\n}\n\nreturn {\n\tresultType : \"pause\"\n};\n","result":""}}]}}
Initialize, Receive, Result
特に書くことはありません。
Execute
wait().then((v) => {
log("wait " + v + " millisecond");
callbackSuccess( {
resultType : "continue"
} );
});
function sleep() {
return new Promise(resolve => setTimeout(resolve,properties.millisecond));
}
async function wait() {
//非同期処理が完了するまで待機
await sleep();
return properties.millisecond.toString();
}
return {
resultType : "pause"
};
失敗した実装
単純に開始時間を保存しておいて、開始時刻から指定秒数経ってwhile文を抜けるsleepの方法です。
ビジーウェイト(Busy Waiting)させるため、並列で動いている他のタグの処理にも影響してsleepさせてしまった失敗した実装方法です。備忘録のために書き残しています。
Initialize, Receive, Result
特に書くことはありません。
Execute
wait();
function wait(){
var time1 = new Date().getTime();
var time2 = new Date().getTime();
while ((time2 - time1)<properties.millisecond){
time2 = new Date().getTime();
}
}
感想
細かくタイミングをずらしたい事が多々あったので、地味でニッチなカスタムタグだけど個人的には凄い便利でした!