John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2014 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 17 | #include "Animator.h" |
| 18 | |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 19 | #include <inttypes.h> |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 20 | #include <set> |
| 21 | |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 22 | #include "AnimationContext.h" |
Tom Hudson | 2dc236b | 2014-10-15 15:46:42 -0400 | [diff] [blame] | 23 | #include "Interpolator.h" |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 24 | #include "RenderNode.h" |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 25 | #include "RenderProperties.h" |
| 26 | |
| 27 | namespace android { |
| 28 | namespace uirenderer { |
| 29 | |
| 30 | /************************************************************ |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 31 | * BaseRenderNodeAnimator |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 32 | ************************************************************/ |
| 33 | |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 34 | BaseRenderNodeAnimator::BaseRenderNodeAnimator(float finalValue) |
Chris Craik | d41c4d8 | 2015-01-05 15:51:13 -0800 | [diff] [blame] | 35 | : mTarget(nullptr) |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 36 | , mStagingTarget(nullptr) |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 37 | , mFinalValue(finalValue) |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 38 | , mDeltaValue(0) |
| 39 | , mFromValue(0) |
Chris Craik | b9ce116d | 2015-08-20 15:14:06 -0700 | [diff] [blame] | 40 | , mStagingPlayState(PlayState::NotStarted) |
| 41 | , mPlayState(PlayState::NotStarted) |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 42 | , mHasStartValue(false) |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 43 | , mStartTime(0) |
Alan Viverette | ad2f8e3 | 2014-05-16 13:28:33 -0700 | [diff] [blame] | 44 | , mDuration(300) |
Chris Craik | 572d9ac | 2014-09-12 17:40:20 -0700 | [diff] [blame] | 45 | , mStartDelay(0) |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 46 | , mMayRunAsync(true) |
| 47 | , mPlayTime(0) { |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 48 | } |
| 49 | |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 50 | BaseRenderNodeAnimator::~BaseRenderNodeAnimator() { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 51 | } |
| 52 | |
| 53 | void BaseRenderNodeAnimator::checkMutable() { |
| 54 | // Should be impossible to hit as the Java-side also has guards for this |
Chris Craik | b9ce116d | 2015-08-20 15:14:06 -0700 | [diff] [blame] | 55 | LOG_ALWAYS_FATAL_IF(mStagingPlayState != PlayState::NotStarted, |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 56 | "Animator has already been started!"); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 57 | } |
| 58 | |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 59 | void BaseRenderNodeAnimator::setInterpolator(Interpolator* interpolator) { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 60 | checkMutable(); |
Chris Craik | 51d6a3d | 2014-12-22 17:16:56 -0800 | [diff] [blame] | 61 | mInterpolator.reset(interpolator); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 62 | } |
| 63 | |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 64 | void BaseRenderNodeAnimator::setStartValue(float value) { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 65 | checkMutable(); |
| 66 | doSetStartValue(value); |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 67 | } |
| 68 | |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 69 | void BaseRenderNodeAnimator::doSetStartValue(float value) { |
| 70 | mFromValue = value; |
| 71 | mDeltaValue = (mFinalValue - mFromValue); |
| 72 | mHasStartValue = true; |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 73 | } |
| 74 | |
Alan Viverette | ad2f8e3 | 2014-05-16 13:28:33 -0700 | [diff] [blame] | 75 | void BaseRenderNodeAnimator::setDuration(nsecs_t duration) { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 76 | checkMutable(); |
Alan Viverette | ad2f8e3 | 2014-05-16 13:28:33 -0700 | [diff] [blame] | 77 | mDuration = duration; |
| 78 | } |
| 79 | |
| 80 | void BaseRenderNodeAnimator::setStartDelay(nsecs_t startDelay) { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 81 | checkMutable(); |
Alan Viverette | ad2f8e3 | 2014-05-16 13:28:33 -0700 | [diff] [blame] | 82 | mStartDelay = startDelay; |
| 83 | } |
| 84 | |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 85 | void BaseRenderNodeAnimator::attach(RenderNode* target) { |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 86 | mStagingTarget = target; |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 87 | onAttached(); |
| 88 | } |
| 89 | |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 90 | void BaseRenderNodeAnimator::start() { |
| 91 | mStagingPlayState = PlayState::Running; |
| 92 | mStagingRequests.push_back(Request::Start); |
| 93 | onStagingPlayStateChanged(); |
| 94 | } |
| 95 | |
| 96 | void BaseRenderNodeAnimator::cancel() { |
| 97 | mStagingPlayState = PlayState::Finished; |
| 98 | mStagingRequests.push_back(Request::Cancel); |
| 99 | onStagingPlayStateChanged(); |
| 100 | } |
| 101 | |
| 102 | void BaseRenderNodeAnimator::reset() { |
| 103 | mStagingPlayState = PlayState::Finished; |
| 104 | mStagingRequests.push_back(Request::Reset); |
| 105 | onStagingPlayStateChanged(); |
| 106 | } |
| 107 | |
| 108 | void BaseRenderNodeAnimator::reverse() { |
| 109 | mStagingPlayState = PlayState::Reversing; |
| 110 | mStagingRequests.push_back(Request::Reverse); |
| 111 | onStagingPlayStateChanged(); |
| 112 | } |
| 113 | |
| 114 | void BaseRenderNodeAnimator::end() { |
| 115 | mStagingPlayState = PlayState::Finished; |
| 116 | mStagingRequests.push_back(Request::End); |
| 117 | onStagingPlayStateChanged(); |
| 118 | } |
| 119 | |
| 120 | void BaseRenderNodeAnimator::resolveStagingRequest(Request request) { |
| 121 | switch (request) { |
| 122 | case Request::Start: |
| 123 | mPlayTime = (mPlayState == PlayState::Running || mPlayState == PlayState::Reversing) ? |
| 124 | mPlayTime : 0; |
| 125 | mPlayState = PlayState::Running; |
Doris Liu | 6725d58 | 2016-08-04 13:20:17 -0700 | [diff] [blame] | 126 | mPendingActionUponFinish = Action::None; |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 127 | break; |
| 128 | case Request::Reverse: |
| 129 | mPlayTime = (mPlayState == PlayState::Running || mPlayState == PlayState::Reversing) ? |
| 130 | mPlayTime : mDuration; |
| 131 | mPlayState = PlayState::Reversing; |
Doris Liu | 6725d58 | 2016-08-04 13:20:17 -0700 | [diff] [blame] | 132 | mPendingActionUponFinish = Action::None; |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 133 | break; |
| 134 | case Request::Reset: |
| 135 | mPlayTime = 0; |
| 136 | mPlayState = PlayState::Finished; |
Doris Liu | 6725d58 | 2016-08-04 13:20:17 -0700 | [diff] [blame] | 137 | mPendingActionUponFinish = Action::Reset; |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 138 | break; |
| 139 | case Request::Cancel: |
| 140 | mPlayState = PlayState::Finished; |
Doris Liu | 6725d58 | 2016-08-04 13:20:17 -0700 | [diff] [blame] | 141 | mPendingActionUponFinish = Action::None; |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 142 | break; |
| 143 | case Request::End: |
| 144 | mPlayTime = mPlayState == PlayState::Reversing ? 0 : mDuration; |
| 145 | mPlayState = PlayState::Finished; |
Doris Liu | 6725d58 | 2016-08-04 13:20:17 -0700 | [diff] [blame] | 146 | mPendingActionUponFinish = Action::End; |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 147 | break; |
| 148 | default: |
| 149 | LOG_ALWAYS_FATAL("Invalid staging request: %d", static_cast<int>(request)); |
| 150 | }; |
| 151 | } |
| 152 | |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 153 | void BaseRenderNodeAnimator::pushStaging(AnimationContext& context) { |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 154 | if (mStagingTarget) { |
| 155 | RenderNode* oldTarget = mTarget; |
| 156 | mTarget = mStagingTarget; |
| 157 | mStagingTarget = nullptr; |
| 158 | if (oldTarget && oldTarget != mTarget) { |
| 159 | oldTarget->onAnimatorTargetChanged(this); |
| 160 | } |
| 161 | } |
| 162 | |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 163 | if (!mHasStartValue) { |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 164 | doSetStartValue(getValue(mTarget)); |
Alan Viverette | ad2f8e3 | 2014-05-16 13:28:33 -0700 | [diff] [blame] | 165 | } |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 166 | |
| 167 | if (!mStagingRequests.empty()) { |
Doris Liu | 148f57f | 2016-02-19 17:19:24 -0800 | [diff] [blame] | 168 | // No interpolator was set, use the default |
| 169 | if (mPlayState == PlayState::NotStarted && !mInterpolator) { |
| 170 | mInterpolator.reset(Interpolator::createDefaultInterpolator()); |
| 171 | } |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 172 | // Keep track of the play state and play time before they are changed when |
| 173 | // staging requests are resolved. |
| 174 | nsecs_t currentPlayTime = mPlayTime; |
| 175 | PlayState prevFramePlayState = mPlayState; |
| 176 | |
| 177 | // Resolve staging requests one by one. |
| 178 | for (Request request : mStagingRequests) { |
| 179 | resolveStagingRequest(request); |
Doris Liu | 766431a | 2016-02-04 22:17:11 +0000 | [diff] [blame] | 180 | } |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 181 | mStagingRequests.clear(); |
| 182 | |
| 183 | if (mStagingPlayState == PlayState::Finished) { |
John Reck | 4d2c472 | 2014-08-29 10:40:56 -0700 | [diff] [blame] | 184 | callOnFinishedListener(context); |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 185 | } else if (mStagingPlayState == PlayState::Running |
| 186 | || mStagingPlayState == PlayState::Reversing) { |
| 187 | bool changed = currentPlayTime != mPlayTime || prevFramePlayState != mStagingPlayState; |
| 188 | if (prevFramePlayState != mStagingPlayState) { |
| 189 | transitionToRunning(context); |
| 190 | } |
| 191 | if (changed) { |
| 192 | // Now we need to seek to the stagingPlayTime (i.e. the animation progress that was |
| 193 | // requested from UI thread). It is achieved by modifying mStartTime, such that |
| 194 | // current time - mStartTime = stagingPlayTime (or mDuration -stagingPlayTime in the |
| 195 | // case of reversing) |
| 196 | nsecs_t currentFrameTime = context.frameTimeMs(); |
| 197 | if (mPlayState == PlayState::Reversing) { |
| 198 | // Reverse is not supported for animations with a start delay, so here we |
| 199 | // assume no start delay. |
| 200 | mStartTime = currentFrameTime - (mDuration - mPlayTime); |
| 201 | } else { |
| 202 | // Animation should play forward |
| 203 | if (mPlayTime == 0) { |
| 204 | // If the request is to start from the beginning, include start delay. |
| 205 | mStartTime = currentFrameTime + mStartDelay; |
| 206 | } else { |
| 207 | // If the request is to seek to a non-zero play time, then we skip start |
| 208 | // delay. |
| 209 | mStartTime = currentFrameTime - mPlayTime; |
| 210 | } |
| 211 | } |
| 212 | } |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 213 | } |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 214 | } |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 215 | onPushStaging(); |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 216 | } |
| 217 | |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 218 | void BaseRenderNodeAnimator::transitionToRunning(AnimationContext& context) { |
| 219 | nsecs_t frameTimeMs = context.frameTimeMs(); |
| 220 | LOG_ALWAYS_FATAL_IF(frameTimeMs <= 0, "%" PRId64 " isn't a real frame time!", frameTimeMs); |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 221 | if (mStartDelay < 0 || mStartDelay > 50000) { |
| 222 | ALOGW("Your start delay is strange and confusing: %" PRId64, mStartDelay); |
| 223 | } |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 224 | mStartTime = frameTimeMs + mStartDelay; |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 225 | if (mStartTime < 0) { |
| 226 | ALOGW("Ended up with a really weird start time of %" PRId64 |
| 227 | " with frame time %" PRId64 " and start delay %" PRId64, |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 228 | mStartTime, frameTimeMs, mStartDelay); |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 229 | // Set to 0 so that the animate() basically instantly finishes |
| 230 | mStartTime = 0; |
| 231 | } |
Doris Liu | 952670d | 2016-04-12 17:43:00 -0700 | [diff] [blame] | 232 | if (mDuration < 0) { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 233 | ALOGW("Your duration is strange and confusing: %" PRId64, mDuration); |
| 234 | } |
| 235 | } |
| 236 | |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 237 | bool BaseRenderNodeAnimator::animate(AnimationContext& context) { |
Chris Craik | b9ce116d | 2015-08-20 15:14:06 -0700 | [diff] [blame] | 238 | if (mPlayState < PlayState::Running) { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 239 | return false; |
| 240 | } |
Chris Craik | b9ce116d | 2015-08-20 15:14:06 -0700 | [diff] [blame] | 241 | if (mPlayState == PlayState::Finished) { |
Doris Liu | 6725d58 | 2016-08-04 13:20:17 -0700 | [diff] [blame] | 242 | if (mPendingActionUponFinish == Action::Reset) { |
| 243 | // Skip to start. |
| 244 | updatePlayTime(0); |
| 245 | } else if (mPendingActionUponFinish == Action::End) { |
| 246 | // Skip to end. |
| 247 | updatePlayTime(mDuration); |
| 248 | } |
| 249 | // Reset pending action. |
| 250 | mPendingActionUponFinish = Action ::None; |
John Reck | 32fb630 | 2014-07-07 09:50:32 -0700 | [diff] [blame] | 251 | return true; |
| 252 | } |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 253 | |
Doris Liu | 766431a | 2016-02-04 22:17:11 +0000 | [diff] [blame] | 254 | // This should be set before setValue() so animators can query this time when setValue |
| 255 | // is called. |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 256 | nsecs_t currentPlayTime = context.frameTimeMs() - mStartTime; |
| 257 | bool finished = updatePlayTime(currentPlayTime); |
| 258 | if (finished && mPlayState != PlayState::Finished) { |
| 259 | mPlayState = PlayState::Finished; |
| 260 | callOnFinishedListener(context); |
| 261 | } |
| 262 | return finished; |
| 263 | } |
Doris Liu | 766431a | 2016-02-04 22:17:11 +0000 | [diff] [blame] | 264 | |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 265 | bool BaseRenderNodeAnimator::updatePlayTime(nsecs_t playTime) { |
| 266 | mPlayTime = mPlayState == PlayState::Reversing ? mDuration - playTime : playTime; |
| 267 | onPlayTimeChanged(mPlayTime); |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 268 | // If BaseRenderNodeAnimator is handling the delay (not typical), then |
| 269 | // because the staging properties reflect the final value, we always need |
| 270 | // to call setValue even if the animation isn't yet running or is still |
| 271 | // being delayed as we need to override the staging value |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 272 | if (playTime < 0) { |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 273 | setValue(mTarget, mFromValue); |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 274 | return false; |
| 275 | } |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 276 | |
| 277 | float fraction = 1.0f; |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 278 | if ((mPlayState == PlayState::Running || mPlayState == PlayState::Reversing) && mDuration > 0) { |
| 279 | fraction = mPlayTime / (float) mDuration; |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 280 | } |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 281 | fraction = MathUtils::clamp(fraction, 0.0f, 1.0f); |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 282 | |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 283 | fraction = mInterpolator->interpolate(fraction); |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 284 | setValue(mTarget, mFromValue + (mDeltaValue * fraction)); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 285 | |
Doris Liu | c4bb185 | 2016-02-19 21:39:21 +0000 | [diff] [blame] | 286 | return playTime >= mDuration; |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 287 | } |
| 288 | |
Doris Liu | 718cd3e | 2016-05-17 16:50:31 -0700 | [diff] [blame] | 289 | nsecs_t BaseRenderNodeAnimator::getRemainingPlayTime() { |
| 290 | return mPlayState == PlayState::Reversing ? mPlayTime : mDuration - mPlayTime; |
| 291 | } |
| 292 | |
John Reck | e2478d4 | 2014-09-03 16:46:05 -0700 | [diff] [blame] | 293 | void BaseRenderNodeAnimator::forceEndNow(AnimationContext& context) { |
Chris Craik | b9ce116d | 2015-08-20 15:14:06 -0700 | [diff] [blame] | 294 | if (mPlayState < PlayState::Finished) { |
| 295 | mPlayState = PlayState::Finished; |
John Reck | e2478d4 | 2014-09-03 16:46:05 -0700 | [diff] [blame] | 296 | callOnFinishedListener(context); |
| 297 | } |
| 298 | } |
| 299 | |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 300 | void BaseRenderNodeAnimator::callOnFinishedListener(AnimationContext& context) { |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 301 | if (mListener.get()) { |
John Reck | 119907c | 2014-08-14 09:02:01 -0700 | [diff] [blame] | 302 | context.callOnFinished(this, mListener.get()); |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 303 | } |
| 304 | } |
| 305 | |
| 306 | /************************************************************ |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 307 | * RenderPropertyAnimator |
| 308 | ************************************************************/ |
| 309 | |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 310 | struct RenderPropertyAnimator::PropertyAccessors { |
| 311 | RenderNode::DirtyPropertyMask dirtyMask; |
| 312 | GetFloatProperty getter; |
| 313 | SetFloatProperty setter; |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 314 | }; |
| 315 | |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 316 | // Maps RenderProperty enum to accessors |
| 317 | const RenderPropertyAnimator::PropertyAccessors RenderPropertyAnimator::PROPERTY_ACCESSOR_LUT[] = { |
| 318 | {RenderNode::TRANSLATION_X, &RenderProperties::getTranslationX, &RenderProperties::setTranslationX }, |
| 319 | {RenderNode::TRANSLATION_Y, &RenderProperties::getTranslationY, &RenderProperties::setTranslationY }, |
John Reck | 41282b7 | 2017-01-11 11:26:43 -0800 | [diff] [blame] | 320 | {RenderNode::TRANSLATION_Z, &RenderProperties::getTranslationZ, &RenderProperties::setTranslationZ }, |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 321 | {RenderNode::SCALE_X, &RenderProperties::getScaleX, &RenderProperties::setScaleX }, |
| 322 | {RenderNode::SCALE_Y, &RenderProperties::getScaleY, &RenderProperties::setScaleY }, |
| 323 | {RenderNode::ROTATION, &RenderProperties::getRotation, &RenderProperties::setRotation }, |
| 324 | {RenderNode::ROTATION_X, &RenderProperties::getRotationX, &RenderProperties::setRotationX }, |
| 325 | {RenderNode::ROTATION_Y, &RenderProperties::getRotationY, &RenderProperties::setRotationY }, |
| 326 | {RenderNode::X, &RenderProperties::getX, &RenderProperties::setX }, |
| 327 | {RenderNode::Y, &RenderProperties::getY, &RenderProperties::setY }, |
| 328 | {RenderNode::Z, &RenderProperties::getZ, &RenderProperties::setZ }, |
| 329 | {RenderNode::ALPHA, &RenderProperties::getAlpha, &RenderProperties::setAlpha }, |
| 330 | }; |
| 331 | |
| 332 | RenderPropertyAnimator::RenderPropertyAnimator(RenderProperty property, float finalValue) |
| 333 | : BaseRenderNodeAnimator(finalValue) |
| 334 | , mPropertyAccess(&(PROPERTY_ACCESSOR_LUT[property])) { |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 335 | } |
| 336 | |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 337 | void RenderPropertyAnimator::onAttached() { |
John Reck | 68bfe0a | 2014-06-24 15:34:58 -0700 | [diff] [blame] | 338 | if (!mHasStartValue |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 339 | && mStagingTarget->isPropertyFieldDirty(mPropertyAccess->dirtyMask)) { |
| 340 | setStartValue((mStagingTarget->stagingProperties().*mPropertyAccess->getter)()); |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 341 | } |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 342 | } |
| 343 | |
| 344 | void RenderPropertyAnimator::onStagingPlayStateChanged() { |
Chris Craik | b9ce116d | 2015-08-20 15:14:06 -0700 | [diff] [blame] | 345 | if (mStagingPlayState == PlayState::Running) { |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 346 | if (mStagingTarget) { |
| 347 | (mStagingTarget->mutateStagingProperties().*mPropertyAccess->setter)(finalValue()); |
| 348 | } else { |
| 349 | // In the case of start delay where stagingTarget has been sync'ed over and null'ed |
| 350 | // we delay the properties update to push staging. |
| 351 | mShouldUpdateStagingProperties = true; |
| 352 | } |
Chris Craik | b9ce116d | 2015-08-20 15:14:06 -0700 | [diff] [blame] | 353 | } else if (mStagingPlayState == PlayState::Finished) { |
John Reck | 32fb630 | 2014-07-07 09:50:32 -0700 | [diff] [blame] | 354 | // We're being canceled, so make sure that whatever values the UI thread |
| 355 | // is observing for us is pushed over |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 356 | mShouldSyncPropertyFields = true; |
| 357 | } |
| 358 | } |
| 359 | |
| 360 | void RenderPropertyAnimator::onPushStaging() { |
| 361 | if (mShouldUpdateStagingProperties) { |
| 362 | (mTarget->mutateStagingProperties().*mPropertyAccess->setter)(finalValue()); |
| 363 | mShouldUpdateStagingProperties = false; |
| 364 | } |
| 365 | |
| 366 | if (mShouldSyncPropertyFields) { |
John Reck | 32fb630 | 2014-07-07 09:50:32 -0700 | [diff] [blame] | 367 | mTarget->setPropertyFieldsDirty(dirtyMask()); |
Doris Liu | 8b08320 | 2016-02-19 21:46:06 +0000 | [diff] [blame] | 368 | mShouldSyncPropertyFields = false; |
John Reck | 8d8af3c | 2014-07-01 15:23:45 -0700 | [diff] [blame] | 369 | } |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 370 | } |
| 371 | |
John Reck | 2218472 | 2014-06-20 07:19:30 -0700 | [diff] [blame] | 372 | uint32_t RenderPropertyAnimator::dirtyMask() { |
| 373 | return mPropertyAccess->dirtyMask; |
| 374 | } |
| 375 | |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 376 | float RenderPropertyAnimator::getValue(RenderNode* target) const { |
| 377 | return (target->properties().*mPropertyAccess->getter)(); |
| 378 | } |
| 379 | |
| 380 | void RenderPropertyAnimator::setValue(RenderNode* target, float value) { |
| 381 | (target->animatorProperties().*mPropertyAccess->setter)(value); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 382 | } |
| 383 | |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 384 | /************************************************************ |
| 385 | * CanvasPropertyPrimitiveAnimator |
| 386 | ************************************************************/ |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 387 | |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 388 | CanvasPropertyPrimitiveAnimator::CanvasPropertyPrimitiveAnimator( |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 389 | CanvasPropertyPrimitive* property, float finalValue) |
| 390 | : BaseRenderNodeAnimator(finalValue) |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 391 | , mProperty(property) { |
| 392 | } |
| 393 | |
Andreas Gampe | 64bb413 | 2014-11-22 00:35:09 +0000 | [diff] [blame] | 394 | float CanvasPropertyPrimitiveAnimator::getValue(RenderNode* target) const { |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 395 | return mProperty->value; |
| 396 | } |
| 397 | |
Andreas Gampe | 64bb413 | 2014-11-22 00:35:09 +0000 | [diff] [blame] | 398 | void CanvasPropertyPrimitiveAnimator::setValue(RenderNode* target, float value) { |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 399 | mProperty->value = value; |
| 400 | } |
| 401 | |
John Reck | a7c2ea2 | 2014-08-08 13:21:00 -0700 | [diff] [blame] | 402 | uint32_t CanvasPropertyPrimitiveAnimator::dirtyMask() { |
| 403 | return RenderNode::DISPLAY_LIST; |
| 404 | } |
| 405 | |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 406 | /************************************************************ |
| 407 | * CanvasPropertySkPaintAnimator |
| 408 | ************************************************************/ |
| 409 | |
| 410 | CanvasPropertyPaintAnimator::CanvasPropertyPaintAnimator( |
John Reck | ff941dc | 2014-05-14 16:34:14 -0700 | [diff] [blame] | 411 | CanvasPropertyPaint* property, PaintField field, float finalValue) |
| 412 | : BaseRenderNodeAnimator(finalValue) |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 413 | , mProperty(property) |
| 414 | , mField(field) { |
| 415 | } |
| 416 | |
Andreas Gampe | 64bb413 | 2014-11-22 00:35:09 +0000 | [diff] [blame] | 417 | float CanvasPropertyPaintAnimator::getValue(RenderNode* target) const { |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 418 | switch (mField) { |
| 419 | case STROKE_WIDTH: |
| 420 | return mProperty->value.getStrokeWidth(); |
| 421 | case ALPHA: |
| 422 | return mProperty->value.getAlpha(); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 423 | } |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 424 | LOG_ALWAYS_FATAL("Unknown field %d", (int) mField); |
| 425 | return -1; |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 426 | } |
| 427 | |
John Reck | 531ee70 | 2014-05-13 10:06:08 -0700 | [diff] [blame] | 428 | static uint8_t to_uint8(float value) { |
| 429 | int c = (int) (value + .5f); |
| 430 | return static_cast<uint8_t>( c < 0 ? 0 : c > 255 ? 255 : c ); |
| 431 | } |
| 432 | |
Andreas Gampe | 64bb413 | 2014-11-22 00:35:09 +0000 | [diff] [blame] | 433 | void CanvasPropertyPaintAnimator::setValue(RenderNode* target, float value) { |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 434 | switch (mField) { |
| 435 | case STROKE_WIDTH: |
| 436 | mProperty->value.setStrokeWidth(value); |
| 437 | return; |
| 438 | case ALPHA: |
John Reck | 531ee70 | 2014-05-13 10:06:08 -0700 | [diff] [blame] | 439 | mProperty->value.setAlpha(to_uint8(value)); |
John Reck | 52244ff | 2014-05-01 21:27:37 -0700 | [diff] [blame] | 440 | return; |
| 441 | } |
| 442 | LOG_ALWAYS_FATAL("Unknown field %d", (int) mField); |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 443 | } |
| 444 | |
John Reck | a7c2ea2 | 2014-08-08 13:21:00 -0700 | [diff] [blame] | 445 | uint32_t CanvasPropertyPaintAnimator::dirtyMask() { |
| 446 | return RenderNode::DISPLAY_LIST; |
| 447 | } |
| 448 | |
Chris Craik | af4d04c | 2014-07-29 12:50:14 -0700 | [diff] [blame] | 449 | RevealAnimator::RevealAnimator(int centerX, int centerY, |
John Reck | d3de42c | 2014-07-15 14:29:33 -0700 | [diff] [blame] | 450 | float startValue, float finalValue) |
| 451 | : BaseRenderNodeAnimator(finalValue) |
| 452 | , mCenterX(centerX) |
Chris Craik | af4d04c | 2014-07-29 12:50:14 -0700 | [diff] [blame] | 453 | , mCenterY(centerY) { |
John Reck | d3de42c | 2014-07-15 14:29:33 -0700 | [diff] [blame] | 454 | setStartValue(startValue); |
| 455 | } |
| 456 | |
| 457 | float RevealAnimator::getValue(RenderNode* target) const { |
Chris Craik | af4d04c | 2014-07-29 12:50:14 -0700 | [diff] [blame] | 458 | return target->properties().getRevealClip().getRadius(); |
John Reck | d3de42c | 2014-07-15 14:29:33 -0700 | [diff] [blame] | 459 | } |
| 460 | |
| 461 | void RevealAnimator::setValue(RenderNode* target, float value) { |
Chris Craik | af4d04c | 2014-07-29 12:50:14 -0700 | [diff] [blame] | 462 | target->animatorProperties().mutableRevealClip().set(true, |
John Reck | d3de42c | 2014-07-15 14:29:33 -0700 | [diff] [blame] | 463 | mCenterX, mCenterY, value); |
| 464 | } |
| 465 | |
John Reck | a7c2ea2 | 2014-08-08 13:21:00 -0700 | [diff] [blame] | 466 | uint32_t RevealAnimator::dirtyMask() { |
| 467 | return RenderNode::GENERIC; |
| 468 | } |
| 469 | |
John Reck | e45b1fd | 2014-04-15 09:50:16 -0700 | [diff] [blame] | 470 | } /* namespace uirenderer */ |
| 471 | } /* namespace android */ |