- ベストアンサー
Actionscript3.0 ストップウォッチ
こんにちは Flash Actionscript3.0 についての質問です。 こちらのサイト http://labs.cybridge.jp/2009/10/actionscript30.html のストップウォッチの、 ボタン部分を自作のシンボルボタンに変えたいのですが、 その場合はコードをどのように書けばいいのでしょうか。 スタートボタンのインスタンス名startbtn ストップボタンのインスタンス名stopbtn です。 よろしくお願いします。
- みんなの回答 (3)
- 専門家の回答
質問者が選んだベストアンサー
#1 & #2 です。 #2 のお礼(補足の方ではないです) を読ませていただきましたが, フレームによって違うボタンなどが登場して そのボタンクリックによって別々の動作をさせるのでしたら その フレーム・フレーム… ごとにスクリプトを書いた方が良いと思います。 少なくともドキュメントクラスのスクリプトだけでするのは難しいでしょう。 とりあえずは ステージ上の各フレームに用意するインスタンスを書きます。 「 」内はインスタンス名です。 ◎ フレーム1 に用意するもの ・ボタンインスタンス「start_btn」 ◎ フレーム2 に用意するもの ・ボタンインスタンス「ok_btn」 ・ボタンインスタンス「cancel_btn」 ・ダイナミックテキストフィールド「time_txt」 ※↑これは フレーム3 と 4 でも継続表示させる ◎ フレーム3 に用意するもの ・ダイナミックテキストフィールド「judge_txt」 ※↑これは フレーム4 でも継続表示させる ・ダイナミックテキストフィールド「time_txt」 ※↑これは フレーム2 から継続表示させる ◎ フレーム4 に用意するもの ・ダイナミックテキストフィールド「judge_txt」 ※↑これは フレーム3 から継続表示させる ・ダイナミックテキストフィールド「time_txt」 ※↑これは フレーム2 から継続表示させる ※下の【添付図】をご参考に。 ドキュメントクラスは無くします。 各フレームに書くスクリプト例です。 //---フレーム1-------------------------- //このフレームで再生を停止 stop(); //経過時間を代入する変数の宣言 var old_time:uint; var now_time:uint; //「start_btn」クリック時に 関数 clockStart を実行 start_btn.addEventListener(MouseEvent.CLICK,clockStart); //関数 clockStart の定義 function clockStart(e:MouseEvent) { //変数 old_time にスタート時の時間を代入 old_time = getTimer(); //このイベントリスナーを削除 start_btn.removeEventListener(MouseEvent.CLICK,clockStart); //次のフレーム(フレーム2)で停止 nextFrame(); } //------------------------------------ //---フレーム2-------------------------- //毎フレーム進む時間で 関数 tick を実行 this.addEventListener(Event.ENTER_FRAME,tick); //関数 tick の定義 function tick(e:Event):void { //変数 now_time に今の時間を代入 now_time = getTimer(); //スタート時からの時間差を取得 var mili_sec = now_time - old_time; //~時間の表示文字列を算出~ var s:int = Math.floor(mili_sec / 1000); var minutes:int = Math.floor(s / 60); var seconds:int = s % 60; var mili:int = mili_sec % 1000; //「time_txt」に経過時間を表示 time_txt.text = minutes + ":" + ("0" + seconds).substr(-2,2) +":"+ ("00" + mili).substr(-3,2); } //「ok_btn」クリック時に 関数 okFunc を実行 ok_btn.addEventListener(MouseEvent.CLICK,okFunc); //関数 okFunc の定義 function okFunc(e:MouseEvent) { //このイベントリスナーを削除 ok_btn.removeEventListener(MouseEvent.CLICK,okFunc); //フレーム3 で停止 gotoAndStop(3); } //「cancel_btn」クリック時に 関数 cancelFunc を実行 cancel_btn.addEventListener(MouseEvent.CLICK,cancelFunc); //関数 cancelFunc の定義 function cancelFunc(e:MouseEvent) { //このイベントリスナーを削除 cancel_btn.removeEventListener(MouseEvent.CLICK,cancelFunc); //フレーム4 で停止 gotoAndStop(4); } //------------------------------------ //---フレーム3-------------------------- //毎フレーム進む時間で 関数 tick を実行するのをやめる this.removeEventListener(Event.ENTER_FRAME,tick); //「judge_txt」に 正 を表示 judge_txt.text = "正"; //------------------------------------ //---フレーム4-------------------------- //毎フレーム進む時間で 関数 tick を実行するのをやめる this.removeEventListener(Event.ENTER_FRAME,tick); //「judge_txt」に 誤 を表示 judge_txt.text = "誤"; //------------------------------------ 以上のように 「何がどうなったときに,何をどうするのか。」 ということを具体的にしっかりとイメージして 1つずつ確実に動作するようにスクリプトを考えて行けばできると思います。
その他の回答 (2)
- BlurFiltan
- ベストアンサー率91% (1611/1754)
#1です。 Google検索例だけ書いて放り出すのも何かと思ったので, #1で書いた内容をふまえた上での修正案を書いておきます。 ドキュメントクラスのクラス名は「FlashTest2」で ファイル名は「FlashTest2.as」ということにしておきます。 //------------------------------------ package { import flash.display.Sprite; import flash.display.SimpleButton; import flash.events.Event; import flash.events.MouseEvent; import flash.utils.getTimer; import flash.text.TextField; public class FlashTest2 extends Sprite { public var OldTime:uint; public var NowTime:uint; public var myTextField:TextField = new TextField(); public function FlashTest2() { // timer text myTextField.x = 100; myTextField.y = 70; myTextField.text = "0:00:00"; this.addChild(myTextField); startbtn.addEventListener(MouseEvent.CLICK, ClockStart); stopbtn.addEventListener(MouseEvent.CLICK, ClockStop); } public function Tick(e:Event):void { NowTime = getTimer(); var MiliSeconds = NowTime - OldTime; var s:int = Math.floor(MiliSeconds / 1000); var minutes:int = Math.floor(s / 60); var seconds:int = s % 60; var mili:int = MiliSeconds % 1000; myTextField.text = minutes + ":" + ("0" + seconds).substr(-2,2) +":"+ ("00" + mili).substr(-3,2); } public function ClockStart(e:MouseEvent) { OldTime = getTimer(); this.addEventListener(Event.ENTER_FRAME,Tick); } public function ClockStop(e:MouseEvent) { this.removeEventListener(Event.ENTER_FRAME,Tick); } } } //------------------------------------ もし数時間以上とか 数日間というような長時間を計る場合は getTimer() でも誤差時間が大きくなるので そのような場合は Date クラスの getTime() メソッドで 1970年1月1日0時からのミリ秒数を取得する方法を使う方が良いでしょう。
お礼
丁寧なご解答ありがとうございます。 そんなに誤差が出てしまうのですね...! 時間はまであれば十分なので大丈夫です。 重ね重ね失礼します。 ストップウォッチの機能を利用して、実験データを作成したいのです。 実験の流れは 1フレーム目:[指示文画面]スタートボタンを押すと[実験画面(2フレーム目)]へ移動。 2フレーム目:[実験画面]二択ボタン どちらかを選ぶと[正(3フレーム目)誤(4フレーム目)画面]へ移動。ここでは、青い『キャンセル』ボタンが誤、普通の『確定』ボタンが正の画面へ移動。 3・4フレーム目:[正誤画面]正か誤と、かかった時間の表示 ●スタートボタン押した時から、二択ボタンどちらかを押した時の時間を計測して最後に表示させたい。 フレームを超えて時間計測するのはどうすればいいのでしょうか。 よろしくお願いします。
補足
こんにちは、 出して下さった修正案と BlurFiltanさんがほかで回答しているページ http://okwave.jp/qa/q7054298.html を参考に試行錯誤しながら少し変えてみました。 □□□□□□□□□□□□□□□□□□□□ package { import flash.display.Sprite; import flash.display.SimpleButton; import flash.events.Event; import flash.events.MouseEvent; import flash.utils.getTimer; import flash.text.TextField; public class FlashTest2 extends Sprite { public var OldTime:uint; public var NowTime:uint; public var myTextField:TextField = new TextField(); public function FlashTest2() { // timer text myTextField.x = 100; myTextField.y = 70; myTextField.text = "0:00:00"; this.addChild(myTextField); startbtn.addEventListener(MouseEvent.CLICK, ClockStart); n_kakutei.addEventListener(MouseEvent.CLICK, ClockStop1); b_cancel.addEventListener(MouseEvent.CLICK, ClockStop2); } public function Tick(e:Event):void { NowTime = getTimer(); var MiliSeconds = NowTime - OldTime; var s:int = Math.floor(MiliSeconds / 1000); var minutes:int = Math.floor(s / 60); var seconds:int = s % 60; var mili:int = MiliSeconds % 1000; myTextField.text = minutes + ":" + ("0" + seconds).substr(-2,2) +":"+ ("00" + mili).substr(-3,2); } public function ClockStart(e:MouseEvent) { OldTime = getTimer(); this.addEventListener(Event.ENTER_FRAME,Tick); MovieClip(startbtn.parent).gotoAndStop(2); } public function ClockStop1(e:MouseEvent) { this.removeEventListener(Event.ENTER_FRAME,Tick); MovieClip(n_kakutei.parent).gotoAndStop(3); } public function ClockStop2(e:MouseEvent) { this.removeEventListener(Event.ENTER_FRAME,Tick); MovieClip(b_cancel.parent).gotoAndStop(4); } } } □□□□□□□□□□□□□□□□□□□□□□ しかし下記エラーが出てしまいました。 Error #1034: 強制型変換に失敗しました。 FlashTest2@28460089 を flash.display.MovieClip に変換できません。 at FlashTest2/ClockStop2() 「強制型変換に失敗しました movieclip」などで検索をかけてみたのですが、よくわからなくて...。 初心者すぎてすいません。 どうすればいいのでしょうか?
- BlurFiltan
- ベストアンサー率91% (1611/1754)
まずは, ご質問に "単純に回答するだけ" でしたら 次のようなスクリプトにすれば良いと思います。 変更はごく一部で大半は削除です。 //--------------------------------------- package { import flash.display.Sprite; import flash.display.SimpleButton; import flash.utils.Timer; import flash.events.TimerEvent; import flash.events.MouseEvent; import flash.text.TextField; public class FlashTest extends Sprite { public var MiliSeconds = 0; public var Clock = new Timer(10); public var myTextField:TextField = new TextField(); //↓削除(コメントアウト) //public var startBTN:Sprite = new Sprite(); //public var stopBTN:Sprite = new Sprite(); public function FlashTest() { // timer text myTextField.x = 100; myTextField.y = 70; myTextField.text = "0:00:00"; this.addChild(myTextField); //↓削除(コメントアウト) // // start button //startBTN.graphics.beginFill(0xFFCC00); //startBTN.graphics.drawRect(60, 5, 40, 40); //this.addChild(startBTN); //↓変更 startbtn.addEventListener(MouseEvent.CLICK, ClockStart); //↓削除(コメントアウト) //startBTN.addEventListener(MouseEvent.MOUSE_OVER, mOver); //startBTN.addEventListener(MouseEvent.MOUSE_OUT, mOut); //↓削除(コメントアウト) // // stop button //stopBTN.graphics.beginFill(0xCCFF00); //stopBTN.graphics.drawRect(160, 5, 40, 40); //this.addChild(stopBTN); //↓変更 stopbtn.addEventListener(MouseEvent.CLICK, ClockStop); //↓削除(コメントアウト) //stopBTN.addEventListener(MouseEvent.MOUSE_OVER, mOver); //stopBTN.addEventListener(MouseEvent.MOUSE_OUT, mOut); // clock Clock.addEventListener(TimerEvent.TIMER, this.Tick); } public function Tick(e:TimerEvent):void { MiliSeconds++; var s:int = Math.floor(MiliSeconds / 100); var minutes:int = Math.floor(s / 60); var seconds:int = s % 60; var mili:int = MiliSeconds % 100; myTextField.text = minutes + ":" + (seconds > 9 ? seconds : "0" + seconds) +":"+ (mili > 9 ? mili : "0" + mili); } //↓削除(コメントアウト) //public function mOver( e) { // e.target.alpha = 0.5; //} //↓削除(コメントアウト) //public function mOut( e ) { // e.target.alpha = 1; //} public function ClockStart(e) { MiliSeconds = 0; Clock.start(); } public function ClockStop(e) { Clock.stop(); } } } //--------------------------------------- しかし...ですね, ハッキリ言ってこのスクリプトは超駄作です(根本的に考え方が間違っています)。 Timer系 はものすごく時の刻みがイイカゲンなのです。 参考にされたサイトのストップウォッチや 今回の回答のために実際に作成してみたストップウォッチを動かしてみましたが, 私の環境では,1分間動かしても約35秒くらいの時しか刻みませんでした。 誤差が膨大すぎてストップウォッチとしては全く使い物になりません。 Timer系 は 定期的にイベントを発生させるためのしくみであって 時を計るために用意されたものではありません。 参考にされたページには 次のような一文があります。 > ストップウォッチの作成方法には > enterframeとtimerの2種類があり > ざっと検索してみたところ、enterframe系の > 処理はサンプルソースがたくさんあったのですが、 > timer系で処理しているものがみあたらなかったので そりゃ当たり前です。 "時の計測自体" に Timer系 を使う人などいません。 Timer系 にしても EnterFrame系 にしても それらによるイベント発生はものすごく時間がイイカゲンですから それ自体の動作間隔で時間や時刻を計測することはできないのです。 もし Timer系 を使うにしても それ自体で時を計測するのではなく EnterFrame系 の使い方と同様に getTimer() または new Date() などで, 定期的に時間や時刻を取得する必要があります。 それは面倒ですし, Timer系 で実行タイミングを細かく刻んでも 結局表示の更新はデフォルトでは EnterFrame のタイミングで行われるわけですから無意味です。 したがって "普通" は Timer系 など使わず EnterFrame系 の実行タイミングで時間や時刻を取得するのです。 (多くのサイトでしている "普通" にはちゃんと意味があるのです。) getTimer() で得る時間もイイカゲンと言えばイイカゲンで, あまり長時間は計れませんがこれほど莫大な誤差は出ません。 誤差は現行の 1000分の1 くらいになると思います。 Google検索例「getTimer() AS3 ストップウォッチ」 http://www.google.co.jp/search?q=getTimer%28%29+AS3+%E3%82%B9%E3%83%88%E3%83%83%E3%83%97%E3%82%A6%E3%82%A9%E3%83%83%E3%83%81
お礼
おそくなってしまい申し訳ありません。 確認しました。問題なく操作できました! 丁寧な説明・画像付きでありがとうございます。 すごく分かりやすくて助かりました。 スクリプト見直してひとつひとつ自分でまた噛み砕いてみますね。 本当にありがとうございました!