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

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

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

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

Q&A

解決済

1回答

1108閲覧

関数を実行後に初期状態に戻したい

MycoChild

総合スコア36

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

0グッド

0クリップ

投稿2021/09/28 08:16

編集2021/09/30 06:58

SOS!!
Swift初心者です。
画像UIView(@IBOutlet weak var ImageView: UIImageView!)を点滅させる関数を実行後に、ある条件を満たしたら初期状態に戻したいです。

Swift5

1extension UIView { 2 func gladAnime() { 3 self.alpha = 1.0 4 UIView.animate( 5 withDuration: 0.8, 6 delay: 0.0, 7 options: [.allowUserInteraction, .curveEaseInOut, .autoreverse, .repeat], 8 animations: { self.alpha = 0 }, completion: nil) 9 } 10}

で、ViewController.swift内

Swift5

1func renderer(_ renderer: SCNSceneRenderer, updateAtTime time: TimeInterval) { 2 中略 3 if ある条件{ 4 DispatchQueue.main.async{ 5 self.ImageView.gladAnime() 6 } 7 }else if 別の条件{ 8  ここでもとに戻したい。 9 } 10}

ある条件を達成したら点滅開始するのですが、別の条件で点滅をストップさせたい(つまり初期状態に戻したい)です。
「ここでもとに戻したい。」のところをどうやっても、ひとたびある条件を達成すると永遠に点滅してます。

どうかご教授ください。

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

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

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

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

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

退会済みユーザー

退会済みユーザー

2021/10/02 17:54

回答に反応がありませんがどうかされましたか?質問の意図と異なる回答である場合はコメントでその旨お伝え頂き質問文に意図を加筆頂けると助かります。
guest

回答1

0

ベストアンサー

UIView.animate() のアニメーションは、UIViewlayer に登録されているので、それを削除するには、layerremoveAnimation(forKey:) を使います。

ただし、UIView.animate() の場合はアニメーションを特定するための keyanimations: の内容次第で勝手に決められてしまうのでわかりません。

アニメーションが登録したひとつしかないと分かっている場合や、ほかに登録されているすべてのアニメーションも削除してよいなら、removeAllAnimations() ですべて削除できます。

swift

1// ここでもとに戻したい。 2self.ImageView.layer.removeAllAnimations()

他にもアニメーションを追加していて、この gladAnime() が最後に追加したアニメーションだと分かっている場合、animationKeys() で登録されたアニメーションの key を取得し、最後の key を指定するなどの方法もあります。

swift

1// ここでもとに戻したい。 2if let lastAnimationKey = self.ImageView.layer.animationKeys()?.last { 3 self.ImageView.layer.removeAnimation(forKey: lastAnimationKey) 4}

いずれの方法を採っても、今回の場合は animations: { self.alpha = 0 } のように不透明度を 0 にしたままなので、アニメーションを削除しても不透明度は 0 のままで画像が見えなくなります。completion: で元に戻すコードが必要です。

swift

1..., 2animations: { self.alpha = 0 }, completion: { _ in self.alpha = 1.0 })

ただし、通常は元々のビューの alpha1.0 とは限らないので、そもそもアニメーション開始時に self.alpha = 1.0 としているのが若干おかしいです。元の不透明度を取得し、アニメーション終了時に戻すようにしてはどうでしょうか。

また、もしアニメーション中のビューそのものにタッチしてアニメーションを止めるような設計の場合、alpha 値が 0 に近いと .allowUserInteraction を指定していても反応しませんので、self.alpha の値を 0.02 などにしたほうが良いです(しきい値はよく分かりません。0.01 だと反応しなかったりします)。

swift

1extension UIView { 2 func gladAnime() { 3 let sourceAlpha = self.alpha 4 UIView.animate( 5 withDuration: 0.8, 6 delay: 0.0, 7 options: [.allowUserInteraction, .curveEaseInOut, .autoreverse, .repeat], 8 animations: { self.alpha = 0.02 }, completion: { _ in self.alpha = sourceAlpha }) 9 } 10}

より詳しい話:

アニメーションの実体は layer に追加する CAAnimation です。UIView.animation() で登録したアニメーションを削除しようとしたとき key が分かりませんでしたが、これは layer に手動でアニメーションを追加登録する際に指定するもので、アニメーションを手動で作成すれば任意のキーを使って登録できます。

たとえば、同じようなアニメーションを CAAnimation で作ると次のようになります。

swift

1extension UIView { 2 func gladAnimeByCA() { 3 let animation = CABasicAnimation(keyPath: "opacity") 4 animation.duration = 0.8 5 animation.beginTime = 0.0 6 animation.timingFunction = CAMediaTimingFunction(name: .easeInEaseOut) 7 animation.autoreverses = true 8 animation.repeatCount = .infinity 9 animation.isRemovedOnCompletion = false 10 animation.toValue = 0.0 11 12 self.layer.add(animation, forKey: "myAnimation") // ここのキーを自由に決められる 13 } 14}

この場合は、key が分かっているので特定のアニメーションを指定の key で確実に削除できます。

swift

1self.ImageView.layer.removeAnimation(forKey: "myAnimation")

投稿2021/09/30 20:07

編集2021/09/30 20:29
退会済みユーザー

退会済みユーザー

総合スコア0

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

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

MycoChild

2021/10/05 09:22

気づくのが遅くなりすみません 丁寧な回答ありがとうございます! なんとなく理解できました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問