質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

Processing

Processingは、オープンソースプロジェクトによるCGのためのプログラミング言語です。Javaをベースにしており、グラフィック機能に特化しています。イメージの生成やアニメーションなど、視覚的なフィードバックを簡単に得ることが可能です。

Q&A

解決済

1回答

2334閲覧

javascript (p5.js) アラーム機能の作成

taki_rentaro

総合スコア6

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

Processing

Processingは、オープンソースプロジェクトによるCGのためのプログラミング言語です。Javaをベースにしており、グラフィック機能に特化しています。イメージの生成やアニメーションなど、視覚的なフィードバックを簡単に得ることが可能です。

0グッド

0クリップ

投稿2021/11/18 21:15

編集2021/11/19 00:21

前提・実現したいこと

アラーム機能を実装したいのですが
左ボタンをON状態でアラームON
左ボタンOFF状態でアラームOFF
にしたいのですが、ボタンと連動して音が正しく鳴らない(ON/OFFが実行されない)状態

不慣れな物で分かりにくい説明になってるかもしれませんがご教示頂けると幸いです

既に試した事・確認した事

function draw()内でalarm();を呼び出した場合は動作する(ON/OFFに関わらずアラーム自体が鳴るかを確認)

112行目にalarm();を呼び出した場合特定のフレーム数でのみON/OFFが実行される

112行目にalarmを呼び出した場合、アラームオシレーターはosc.amp(0.05, 0.05);だけが反映される(本来「ピピピ」と細切れな音になるはずが「ピー」という連続音になってしまう)

###該当のソースコード

/* */ let osc; // 生成するオシレータを入れる let b1 = {sw: false, state: false}; // 左のボタンの状態(連想配列) let b2 = {sw_2: false, state_2: false}; // 右のボタンの状態(連想配列) function setup() { createCanvas(600, 600); frameRate(60); osc = new p5.Oscillator(); // 発信機機能を生成 osc.setType('square'); // 矩形波を設定(他の音:triangle,saw,sin) osc.freq(1000); // 周波数 1kHz osc.amp(0); // アンプの音量 音量0 osc.start(); // 発信機スタート } //アナログ時計 function draw() { background(50); push(); translate(width / 2, height / 2); let h = (hour() + minute() / 60) / 12 * TWO_PI; let m = (minute() + second() / 60) / 60 * TWO_PI; let s = second() / 60 * TWO_PI; //外枠 noFill(); stroke(255); strokeWeight(5); circle(0, 0, 445); //文字盤 push();  for (let i = 0; i < 12; i++) {  rotate(TWO_PI / 12);  noStroke();  fill(255);  circle(0, -185, 32);  }  pop(); //時針 strokeWeight(5); draw_hands(100, h); //分針 strokeWeight(3); draw_hands(125, m); //秒針 strokeWeight(1); draw_hands(150, s); //デジタル時計 digital_Time(0,0); pop(); //左ボタン let msg; let sw = button(80, 80, 50, b1) // 指定した場所に指定した大きさのボタンを生成 if (sw) msg = "ON"; else msg = "OFF"; textSize(24); fill(225); text(msg, 60, 45); //右ボタン let msg_2; let sw_2 = button_2(520, 80, 50, b2); if (sw_2) msg_2 = "ON"; else msg_2 = "OFF"; textSize(24); fill(225); text(msg_2, 500, 45); //アラーム alarm(); } function draw_hands(len, angle) { let x = len * sin(angle); let y = len * -cos(angle); stroke(255); line(0, 0, x, y); } //デジタル時計 function digital_Time(x,y) { noStroke(); fill(225); textSize(28); textAlign(CENTER); //translateにより座標(0,0)がcanvasの中心にある text(hour() + ':' + minute() + ':' + second(),0,-50); } // 左ボタンを生成する関数 function button(x, y, size, obj) { // objは状態を記憶する連想配列 sw, state push(); if (mouseIsPressed && dist(x, y, mouseX, mouseY) < size / 2) { if (!obj.state) { // いままでボタンが押されていなかったならば obj.state = true; // ボタンを押されている状態obj.stateをtrueに変更する obj.sw = !obj.sw; // スイッチobj.swを切り換える **//ここにアラーム関数入れる** } }else{ obj.state = false; // ボタンが押されてないので、obj.stateをリセットする } if (obj.sw) fill("red"); // obj.swの内容がtrueならば赤 (ON, true) else fill("blue");// obj.swの内容がfalseならば青(OFF, false) strokeWeight(2.5); ellipse(x, y, size, size); pop(); return obj.sw; // ボタンの状態を返す } // 右ボタンを生成する関数 function button_2(x, y, size, obj) { // objは状態を記憶する連想配列 sw, state push(); if (mouseIsPressed && dist(x, y, mouseX, mouseY) < size / 2) { if (!obj.state) { // いままでボタンが押されていなかったならば obj.state = true; // ボタンを押されている状態obj.stateをtrueに変更する obj.sw_2 = !obj.sw_2; // スイッチobj.swを切り換える } }else{ obj.state = false; // ボタンが押されてないので、obj.stateをリセットする } if (obj.sw_2) fill("red"); // obj.swの内容がtrueならば赤 (ON, true) else fill("blue");// obj.swの内容がfalseならば青(OFF, false) strokeWeight(2.5); ellipse(x, y, size, size); pop(); return obj.sw_2; // ボタンの状態を返す } // アラーム function alarm(){ if (frameCount % 10 == 0) osc.amp(0.05, 0.05); // 0.05秒かけて音量を1.0にする else if (frameCount % 10 == 5) osc.amp(0, 0.05); // 0.05秒かけて音量を0にする }

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

112行目にalarm();を呼び出した場合特定のフレーム数でのみON/OFFが実行される
112行目にalarmを呼び出した場合、アラームオシレーターはosc.amp(0.05, 0.05);だけが反映される

「//ここにアラーム関数入れる」この行のことですよね?

ここに入るのはボタンを押した一瞬だけです(高速でON・OFFを繰り返さないようにstateを作ったんですよね?)
その瞬間がframeCount % 10 == 0なら音が鳴りますし(「ピー」)、そうでないなら鳴りません。
当然「ピピピ」にもならないです。

alarm()を入れるなら、if (obj.sw)のほうじゃないですか?

js

1// 左ボタンを生成する関数 2function button(x, y, size, obj) { 3 push(); 4 if (mouseIsPressed && dist(x, y, mouseX, mouseY) < size / 2) { 5 if (!obj.state) { 6 obj.state = true; 7 obj.sw = !obj.sw; 8 } 9 } else { 10 obj.state = false; 11 } 12 13 if (obj.sw) { 14 fill("red"); 15 alarm(); 16 } else { 17 fill("blue"); 18 osc.amp(0, 0.05); 19 } 20 strokeWeight(2.5); 21 ellipse(x, y, size, size); 22 pop(); 23 return obj.sw; 24}

以下勝手に改良^^;

この場合はmouseIsPressedより、mousePressed()のほうが扱いやすいと思います(押したときに1回だけ呼ばれるので)
reference | mousePressed()

Oscillatorが動き続けている?のも何となく気持ち悪いです(明示的にstartstopしたほうが意図が分かりやすくないですか)

button_2()は明らかに重複コードですよね(どうまとめるかですが、b1b2を拡張しました。クラスにしてもいいかもしれません)

js

1// コンストラクタで指定できます new p5.Oscillator([freq], [type]) 2const osc = new p5.Oscillator(1000, 'square'); 3 4// mousePressed()でstateがいらなくなった buttonを共通化するためx, y, sizeを追加 5const b1 = { x:80, y:80, size:50, sw:false }; 6const b2 = { x:520, y:80, size:50, sw:false }; 7 8function setup() { 9 createCanvas(600, 600); 10 frameRate(60); 11} 12 13function draw() { 14 background(50); 15 16 push(); 17 translate(width / 2, height / 2); 18 noFill(); 19 stroke(255); 20 strokeWeight(5); 21 circle(0, 0, 445); 22 23 push(); 24 for (let i = 0; i < 12; i++) { 25 rotate(TWO_PI / 12); 26 noStroke(); 27 fill(255); 28 circle(0, -185, 32); 29 } 30 pop(); 31 32 const h = (hour() + minute() / 60) / 12 * TWO_PI; 33 const m = (minute() + second() / 60) / 60 * TWO_PI; 34 const s = second() / 60 * TWO_PI; 35 drawHands(100, h, 5); // strokeWeightはdrawHands内に 36 drawHands(125, m, 3); 37 drawHands(150, s, 1); 38 39 digitalTime(0, -50); // 引数を渡すならちゃんと使ったほうがw 40 pop(); 41 42 toggleButton(b1); 43 toggleButton(b2); 44 45 if (b1.sw) { 46 alarm(); 47 } 48} 49 50function drawHands(len, angle, weight) { 51 const x = len * sin(angle); 52 const y = len * -cos(angle); 53 stroke(255); 54 strokeWeight(weight); 55 line(0, 0, x, y); 56} 57 58function digitalTime(x, y) { 59 noStroke(); 60 fill(225); 61 textSize(28); 62 textAlign(CENTER); 63 64 // nfは1を01にします nf(値, [整数部の桁数], [少数部の桁数]) 65 text(nf(hour(), 2) + ':' + nf(minute(), 2) + ':' + nf(second(), 2), x, y); 66} 67 68function toggleButton(obj) { 69 fill(obj.sw ? 'red' : 'blue'); // 条件演算子(3項演算子)でスッキリします 70 strokeWeight(2.5); 71 circle(obj.x, obj.y, obj.size); 72 73 textSize(24); 74 fill(225); 75 text(obj.sw ? 'ON' : 'OFF', obj.x - 20, obj.y - 35); 76} 77 78function alarm() { 79 if (frameCount % 10 == 0) { 80 osc.amp(0.05, 0.05); 81 } else if (frameCount % 10 == 5) { 82 osc.amp(0, 0.05); 83 } 84} 85 86function mousePressed() { // マウスを押したとき... 87 if (dist(b1.x, b1.y, mouseX, mouseY) < b1.size / 2) { 88 b1.sw = !b1.sw; 89 90 if (b1.sw) { 91 osc.start(); 92 } else { 93 osc.stop(); 94 } 95 } 96 97 if (dist(b2.x, b2.y, mouseX, mouseY) < b2.size / 2) { 98 b2.sw = !b2.sw; 99 } 100}

投稿2021/11/19 09:25

編集2023/07/29 08:39
TN8001

総合スコア9326

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問