| OLD | NEW |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "ui/events/gesture_detection/motion_event_buffer.h" | 5 #include "ui/events/gesture_detection/motion_event_buffer.h" |
| 6 | 6 |
| 7 #include "base/debug/trace_event.h" | 7 #include "base/debug/trace_event.h" |
| 8 #include "ui/events/gesture_detection/motion_event.h" | |
| 9 #include "ui/events/gesture_detection/motion_event_generic.h" | 8 #include "ui/events/gesture_detection/motion_event_generic.h" |
| 10 | 9 |
| 11 namespace ui { | 10 namespace ui { |
| 12 namespace { | 11 namespace { |
| 13 | 12 |
| 14 // Latency added during resampling. A few milliseconds doesn't hurt much but | 13 // Latency added during resampling. A few milliseconds doesn't hurt much but |
| 15 // reduces the impact of mispredicted touch positions. | 14 // reduces the impact of mispredicted touch positions. |
| 16 const int kResampleLatencyMs = 5; | 15 const int kResampleLatencyMs = 5; |
| 17 | 16 |
| 18 // Minimum time difference between consecutive samples before attempting to | 17 // Minimum time difference between consecutive samples before attempting to |
| 19 // resample. | 18 // resample. |
| 20 const int kResampleMinDeltaMs = 2; | 19 const int kResampleMinDeltaMs = 2; |
| 21 | 20 |
| 22 // Maximum time to predict forward from the last known state, to avoid | 21 // Maximum time to predict forward from the last known state, to avoid |
| 23 // predicting too far into the future. This time is further bounded by 50% of | 22 // predicting too far into the future. This time is further bounded by 50% of |
| 24 // the last time delta. | 23 // the last time delta. |
| 25 const int kResampleMaxPredictionMs = 8; | 24 const int kResampleMaxPredictionMs = 8; |
| 26 | 25 |
| 27 typedef ScopedVector<MotionEvent> MotionEventVector; | 26 typedef ScopedVector<MotionEventGeneric> MotionEventVector; |
| 28 | 27 |
| 29 float Lerp(float a, float b, float alpha) { | 28 float Lerp(float a, float b, float alpha) { |
| 30 return a + alpha * (b - a); | 29 return a + alpha * (b - a); |
| 31 } | 30 } |
| 32 | 31 |
| 33 bool CanAddSample(const MotionEvent& event0, const MotionEvent& event1) { | 32 bool CanAddSample(const MotionEvent& event0, const MotionEvent& event1) { |
| 34 DCHECK_EQ(event0.GetAction(), MotionEvent::ACTION_MOVE); | 33 DCHECK_EQ(event0.GetAction(), MotionEvent::ACTION_MOVE); |
| 35 if (event1.GetAction() != MotionEvent::ACTION_MOVE) | 34 if (event1.GetAction() != MotionEvent::ACTION_MOVE) |
| 36 return false; | 35 return false; |
| 37 | 36 |
| (...skipping 41 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 79 MotionEventVector unconsumed_batch; | 78 MotionEventVector unconsumed_batch; |
| 80 unconsumed_batch.insert( | 79 unconsumed_batch.insert( |
| 81 unconsumed_batch.begin(), batch->begin() + count, batch->end()); | 80 unconsumed_batch.begin(), batch->begin() + count, batch->end()); |
| 82 batch->weak_erase(batch->begin() + count, batch->end()); | 81 batch->weak_erase(batch->begin() + count, batch->end()); |
| 83 | 82 |
| 84 unconsumed_batch.swap(*batch); | 83 unconsumed_batch.swap(*batch); |
| 85 DCHECK_GE(unconsumed_batch.size(), 1U); | 84 DCHECK_GE(unconsumed_batch.size(), 1U); |
| 86 return unconsumed_batch.Pass(); | 85 return unconsumed_batch.Pass(); |
| 87 } | 86 } |
| 88 | 87 |
| 89 PointerProperties PointerFromMotionEvent(const MotionEvent& event, | 88 // Linearly interpolate the pointer position between two MotionEvent samples. |
| 90 size_t pointer_index) { | 89 // Only pointers of finger or unknown type will be resampled. |
| 91 PointerProperties result; | |
| 92 result.id = event.GetPointerId(pointer_index); | |
| 93 result.tool_type = event.GetToolType(pointer_index); | |
| 94 result.x = event.GetX(pointer_index); | |
| 95 result.y = event.GetY(pointer_index); | |
| 96 result.raw_x = event.GetRawX(pointer_index); | |
| 97 result.raw_y = event.GetRawY(pointer_index); | |
| 98 result.pressure = event.GetPressure(pointer_index); | |
| 99 result.touch_major = event.GetTouchMajor(pointer_index); | |
| 100 result.touch_minor = event.GetTouchMinor(pointer_index); | |
| 101 result.orientation = event.GetOrientation(pointer_index); | |
| 102 return result; | |
| 103 } | |
| 104 | |
| 105 PointerProperties ResamplePointer(const MotionEvent& event0, | 90 PointerProperties ResamplePointer(const MotionEvent& event0, |
| 106 const MotionEvent& event1, | 91 const MotionEvent& event1, |
| 107 size_t event0_pointer_index, | 92 size_t event0_pointer_index, |
| 108 size_t event1_pointer_index, | 93 size_t event1_pointer_index, |
| 109 float alpha) { | 94 float alpha) { |
| 110 DCHECK_EQ(event0.GetPointerId(event0_pointer_index), | 95 DCHECK_EQ(event0.GetPointerId(event0_pointer_index), |
| 111 event1.GetPointerId(event1_pointer_index)); | 96 event1.GetPointerId(event1_pointer_index)); |
| 112 // If the tool should not be resampled, use the latest event in the valid | 97 // If the tool should not be resampled, use the latest event in the valid |
| 113 // horizon (i.e., the event no later than the time interpolated by alpha). | 98 // horizon (i.e., the event no later than the time interpolated by alpha). |
| 114 if (!ShouldResampleTool(event0.GetToolType(event0_pointer_index))) { | 99 if (!ShouldResampleTool(event0.GetToolType(event0_pointer_index))) { |
| 115 if (alpha > 1) | 100 if (alpha > 1) |
| 116 return PointerFromMotionEvent(event1, event1_pointer_index); | 101 return PointerProperties(event1, event1_pointer_index); |
| 117 else | 102 else |
| 118 return PointerFromMotionEvent(event0, event0_pointer_index); | 103 return PointerProperties(event0, event0_pointer_index); |
| 119 } | 104 } |
| 120 | 105 |
| 121 PointerProperties p(PointerFromMotionEvent(event0, event0_pointer_index)); | 106 PointerProperties p(event0, event0_pointer_index); |
| 122 p.x = Lerp(p.x, event1.GetX(event1_pointer_index), alpha); | 107 p.x = Lerp(p.x, event1.GetX(event1_pointer_index), alpha); |
| 123 p.y = Lerp(p.y, event1.GetY(event1_pointer_index), alpha); | 108 p.y = Lerp(p.y, event1.GetY(event1_pointer_index), alpha); |
| 124 p.raw_x = Lerp(p.raw_x, event1.GetRawX(event1_pointer_index), alpha); | 109 p.raw_x = Lerp(p.raw_x, event1.GetRawX(event1_pointer_index), alpha); |
| 125 p.raw_y = Lerp(p.raw_y, event1.GetRawY(event1_pointer_index), alpha); | 110 p.raw_y = Lerp(p.raw_y, event1.GetRawY(event1_pointer_index), alpha); |
| 126 return p; | 111 return p; |
| 127 } | 112 } |
| 128 | 113 |
| 129 scoped_ptr<MotionEvent> ResampleMotionEvent(const MotionEvent& event0, | 114 // Linearly interpolate the pointers between two event samples using the |
| 130 const MotionEvent& event1, | 115 // provided |resample_time|. |
| 131 base::TimeTicks resample_time) { | 116 scoped_ptr<MotionEventGeneric> ResampleMotionEvent( |
| 117 const MotionEvent& event0, |
| 118 const MotionEvent& event1, |
| 119 base::TimeTicks resample_time) { |
| 132 DCHECK_EQ(MotionEvent::ACTION_MOVE, event0.GetAction()); | 120 DCHECK_EQ(MotionEvent::ACTION_MOVE, event0.GetAction()); |
| 133 DCHECK_EQ(event0.GetPointerCount(), event1.GetPointerCount()); | 121 DCHECK_EQ(event0.GetPointerCount(), event1.GetPointerCount()); |
| 134 | 122 |
| 135 const base::TimeTicks time0 = event0.GetEventTime(); | 123 const base::TimeTicks time0 = event0.GetEventTime(); |
| 136 const base::TimeTicks time1 = event1.GetEventTime(); | 124 const base::TimeTicks time1 = event1.GetEventTime(); |
| 137 DCHECK(time0 < time1); | 125 DCHECK(time0 < time1); |
| 138 DCHECK(time0 <= resample_time); | 126 DCHECK(time0 <= resample_time); |
| 139 | 127 |
| 140 const float alpha = (resample_time - time0).InMillisecondsF() / | 128 const float alpha = (resample_time - time0).InMillisecondsF() / |
| 141 (time1 - time0).InMillisecondsF(); | 129 (time1 - time0).InMillisecondsF(); |
| (...skipping 12 matching lines...) Expand all Loading... |
| 154 MotionEvent::ACTION_MOVE, resample_time, pointer)); | 142 MotionEvent::ACTION_MOVE, resample_time, pointer)); |
| 155 } else { | 143 } else { |
| 156 event->PushPointer(pointer); | 144 event->PushPointer(pointer); |
| 157 } | 145 } |
| 158 } | 146 } |
| 159 | 147 |
| 160 DCHECK(event); | 148 DCHECK(event); |
| 161 event->set_id(event0.GetId()); | 149 event->set_id(event0.GetId()); |
| 162 event->set_action_index(event0.GetActionIndex()); | 150 event->set_action_index(event0.GetActionIndex()); |
| 163 event->set_button_state(event0.GetButtonState()); | 151 event->set_button_state(event0.GetButtonState()); |
| 164 | |
| 165 return event.Pass(); | 152 return event.Pass(); |
| 166 } | 153 } |
| 167 | 154 |
| 168 // MotionEvent implementation for storing multiple events, with the most | 155 // Synthesize a compound MotionEventGeneric event from a sequence of events. |
| 169 // recent event used as the base event, and prior events used as the history. | 156 // Events must be in non-decreasing (time) order. |
| 170 class CompoundMotionEvent : public ui::MotionEvent { | 157 scoped_ptr<MotionEventGeneric> ConsumeSamples(MotionEventVector events) { |
| 171 public: | 158 DCHECK(!events.empty()); |
| 172 explicit CompoundMotionEvent(MotionEventVector events) | 159 scoped_ptr<MotionEventGeneric> event(events.back()); |
| 173 : events_(events.Pass()) { | 160 for (size_t i = 0; i + 1 < events.size(); ++i) |
| 174 DCHECK_GE(events_.size(), 1U); | 161 event->PushHistoricalEvent(scoped_ptr<MotionEvent>(events[i])); |
| 175 } | 162 events.weak_clear(); |
| 176 ~CompoundMotionEvent() override {} | 163 return event.Pass(); |
| 164 } |
| 177 | 165 |
| 178 int GetId() const override { return latest().GetId(); } | 166 // Consume a series of event samples, attempting to synthesize a new, synthetic |
| 167 // event if the samples and sample time meet certain interpolation/extrapolation |
| 168 // conditions. If such conditions are met, the provided samples will be added |
| 169 // to the synthetic event's history, otherwise, the samples will be used to |
| 170 // generate a basic, compound event. |
| 171 // TODO(jdduke): Revisit resampling to handle cases where alternating frames |
| 172 // are resampled or resampling is otherwise inconsistent, e.g., a 90hz input |
| 173 // and 60hz frame signal could phase-align such that even frames yield an |
| 174 // extrapolated event and odd frames are not resampled, crbug.com/399381. |
| 175 scoped_ptr<MotionEventGeneric> ConsumeSamplesAndTryResampling( |
| 176 base::TimeTicks resample_time, |
| 177 MotionEventVector events, |
| 178 const MotionEvent* next) { |
| 179 const ui::MotionEvent* event0 = nullptr; |
| 180 const ui::MotionEvent* event1 = nullptr; |
| 181 if (next) { |
| 182 DCHECK(resample_time < next->GetEventTime()); |
| 183 // Interpolate between current sample and future sample. |
| 184 event0 = events.back(); |
| 185 event1 = next; |
| 186 } else if (events.size() >= 2) { |
| 187 // Extrapolate future sample using current sample and past sample. |
| 188 event0 = events[events.size() - 2]; |
| 189 event1 = events[events.size() - 1]; |
| 179 | 190 |
| 180 Action GetAction() const override { return latest().GetAction(); } | 191 const base::TimeTicks time1 = event1->GetEventTime(); |
| 181 | 192 base::TimeTicks max_predict = |
| 182 int GetActionIndex() const override { return latest().GetActionIndex(); } | 193 time1 + |
| 183 | 194 std::min((event1->GetEventTime() - event0->GetEventTime()) / 2, |
| 184 size_t GetPointerCount() const override { return latest().GetPointerCount(); } | 195 base::TimeDelta::FromMilliseconds(kResampleMaxPredictionMs)); |
| 185 | 196 if (resample_time > max_predict) { |
| 186 int GetPointerId(size_t pointer_index) const override { | 197 TRACE_EVENT_INSTANT2("input", |
| 187 return latest().GetPointerId(pointer_index); | 198 "MotionEventBuffer::TryResample prediction adjust", |
| 199 TRACE_EVENT_SCOPE_THREAD, |
| 200 "original(ms)", |
| 201 (resample_time - time1).InMilliseconds(), |
| 202 "adjusted(ms)", |
| 203 (max_predict - time1).InMilliseconds()); |
| 204 resample_time = max_predict; |
| 205 } |
| 206 } else { |
| 207 TRACE_EVENT_INSTANT0("input", |
| 208 "MotionEventBuffer::TryResample insufficient data", |
| 209 TRACE_EVENT_SCOPE_THREAD); |
| 210 return ConsumeSamples(events.Pass()); |
| 188 } | 211 } |
| 189 | 212 |
| 190 float GetX(size_t pointer_index) const override { | 213 DCHECK(event0); |
| 191 return latest().GetX(pointer_index); | 214 DCHECK(event1); |
| 215 const base::TimeTicks time0 = event0->GetEventTime(); |
| 216 const base::TimeTicks time1 = event1->GetEventTime(); |
| 217 base::TimeDelta delta = time1 - time0; |
| 218 if (delta < base::TimeDelta::FromMilliseconds(kResampleMinDeltaMs)) { |
| 219 TRACE_EVENT_INSTANT1("input", |
| 220 "MotionEventBuffer::TryResample failure", |
| 221 TRACE_EVENT_SCOPE_THREAD, |
| 222 "event_delta_too_small(ms)", |
| 223 delta.InMilliseconds()); |
| 224 return ConsumeSamples(events.Pass()); |
| 192 } | 225 } |
| 193 | 226 |
| 194 float GetY(size_t pointer_index) const override { | 227 scoped_ptr<MotionEventGeneric> resampled_event = |
| 195 return latest().GetY(pointer_index); | 228 ResampleMotionEvent(*event0, *event1, resample_time); |
| 196 } | 229 for (size_t i = 0; i < events.size(); ++i) |
| 197 | 230 resampled_event->PushHistoricalEvent(scoped_ptr<MotionEvent>(events[i])); |
| 198 float GetRawX(size_t pointer_index) const override { | 231 events.weak_clear(); |
| 199 return latest().GetRawX(pointer_index); | 232 return resampled_event.Pass(); |
| 200 } | 233 } |
| 201 | |
| 202 float GetRawY(size_t pointer_index) const override { | |
| 203 return latest().GetRawY(pointer_index); | |
| 204 } | |
| 205 | |
| 206 float GetTouchMajor(size_t pointer_index) const override { | |
| 207 return latest().GetTouchMajor(pointer_index); | |
| 208 } | |
| 209 | |
| 210 float GetTouchMinor(size_t pointer_index) const override { | |
| 211 return latest().GetTouchMinor(pointer_index); | |
| 212 } | |
| 213 | |
| 214 float GetOrientation(size_t pointer_index) const override { | |
| 215 return latest().GetOrientation(pointer_index); | |
| 216 } | |
| 217 | |
| 218 float GetPressure(size_t pointer_index) const override { | |
| 219 return latest().GetPressure(pointer_index); | |
| 220 } | |
| 221 | |
| 222 ToolType GetToolType(size_t pointer_index) const override { | |
| 223 return latest().GetToolType(pointer_index); | |
| 224 } | |
| 225 | |
| 226 int GetButtonState() const override { return latest().GetButtonState(); } | |
| 227 | |
| 228 int GetFlags() const override { return latest().GetFlags(); } | |
| 229 | |
| 230 base::TimeTicks GetEventTime() const override { | |
| 231 return latest().GetEventTime(); | |
| 232 } | |
| 233 | |
| 234 size_t GetHistorySize() const override { return events_.size() - 1; } | |
| 235 | |
| 236 base::TimeTicks GetHistoricalEventTime( | |
| 237 size_t historical_index) const override { | |
| 238 DCHECK_LT(historical_index, GetHistorySize()); | |
| 239 return events_[historical_index]->GetEventTime(); | |
| 240 } | |
| 241 | |
| 242 float GetHistoricalTouchMajor(size_t pointer_index, | |
| 243 size_t historical_index) const override { | |
| 244 DCHECK_LT(historical_index, GetHistorySize()); | |
| 245 return events_[historical_index]->GetTouchMajor(); | |
| 246 } | |
| 247 | |
| 248 float GetHistoricalX(size_t pointer_index, | |
| 249 size_t historical_index) const override { | |
| 250 DCHECK_LT(historical_index, GetHistorySize()); | |
| 251 return events_[historical_index]->GetX(pointer_index); | |
| 252 } | |
| 253 | |
| 254 float GetHistoricalY(size_t pointer_index, | |
| 255 size_t historical_index) const override { | |
| 256 DCHECK_LT(historical_index, GetHistorySize()); | |
| 257 return events_[historical_index]->GetY(pointer_index); | |
| 258 } | |
| 259 | |
| 260 scoped_ptr<MotionEvent> Clone() const override { | |
| 261 MotionEventVector cloned_events; | |
| 262 cloned_events.reserve(events_.size()); | |
| 263 for (size_t i = 0; i < events_.size(); ++i) | |
| 264 cloned_events.push_back(events_[i]->Clone().release()); | |
| 265 return scoped_ptr<MotionEvent>( | |
| 266 new CompoundMotionEvent(cloned_events.Pass())); | |
| 267 } | |
| 268 | |
| 269 scoped_ptr<MotionEvent> Cancel() const override { return latest().Cancel(); } | |
| 270 | |
| 271 // Returns the new, resampled event, or NULL if none was created. | |
| 272 // TODO(jdduke): Revisit resampling to handle cases where alternating frames | |
| 273 // are resampled or resampling is otherwise inconsistent, e.g., a 90hz input | |
| 274 // and 60hz frame signal could phase-align such that even frames yield an | |
| 275 // extrapolated event and odd frames are not resampled, crbug.com/399381. | |
| 276 const MotionEvent* TryResample(base::TimeTicks resample_time, | |
| 277 const ui::MotionEvent* next) { | |
| 278 DCHECK_EQ(GetAction(), ACTION_MOVE); | |
| 279 const ui::MotionEvent* event0 = NULL; | |
| 280 const ui::MotionEvent* event1 = NULL; | |
| 281 if (next) { | |
| 282 DCHECK(resample_time < next->GetEventTime()); | |
| 283 // Interpolate between current sample and future sample. | |
| 284 event0 = events_.back(); | |
| 285 event1 = next; | |
| 286 } else if (events_.size() >= 2) { | |
| 287 // Extrapolate future sample using current sample and past sample. | |
| 288 event0 = events_[events_.size() - 2]; | |
| 289 event1 = events_[events_.size() - 1]; | |
| 290 | |
| 291 const base::TimeTicks time1 = event1->GetEventTime(); | |
| 292 base::TimeTicks max_predict = | |
| 293 time1 + | |
| 294 std::min((event1->GetEventTime() - event0->GetEventTime()) / 2, | |
| 295 base::TimeDelta::FromMilliseconds(kResampleMaxPredictionMs)); | |
| 296 if (resample_time > max_predict) { | |
| 297 TRACE_EVENT_INSTANT2("input", | |
| 298 "MotionEventBuffer::TryResample prediction adjust", | |
| 299 TRACE_EVENT_SCOPE_THREAD, | |
| 300 "original(ms)", | |
| 301 (resample_time - time1).InMilliseconds(), | |
| 302 "adjusted(ms)", | |
| 303 (max_predict - time1).InMilliseconds()); | |
| 304 resample_time = max_predict; | |
| 305 } | |
| 306 } else { | |
| 307 TRACE_EVENT_INSTANT0("input", | |
| 308 "MotionEventBuffer::TryResample insufficient data", | |
| 309 TRACE_EVENT_SCOPE_THREAD); | |
| 310 return NULL; | |
| 311 } | |
| 312 | |
| 313 DCHECK(event0); | |
| 314 DCHECK(event1); | |
| 315 const base::TimeTicks time0 = event0->GetEventTime(); | |
| 316 const base::TimeTicks time1 = event1->GetEventTime(); | |
| 317 base::TimeDelta delta = time1 - time0; | |
| 318 if (delta < base::TimeDelta::FromMilliseconds(kResampleMinDeltaMs)) { | |
| 319 TRACE_EVENT_INSTANT1("input", | |
| 320 "MotionEventBuffer::TryResample failure", | |
| 321 TRACE_EVENT_SCOPE_THREAD, | |
| 322 "event_delta_too_small(ms)", | |
| 323 delta.InMilliseconds()); | |
| 324 return NULL; | |
| 325 } | |
| 326 | |
| 327 events_.push_back( | |
| 328 ResampleMotionEvent(*event0, *event1, resample_time).release()); | |
| 329 return events_.back(); | |
| 330 } | |
| 331 | |
| 332 size_t samples() const { return events_.size(); } | |
| 333 | |
| 334 private: | |
| 335 const MotionEvent& latest() const { return *events_.back(); } | |
| 336 | |
| 337 // Events are in order from oldest to newest. | |
| 338 MotionEventVector events_; | |
| 339 | |
| 340 DISALLOW_COPY_AND_ASSIGN(CompoundMotionEvent); | |
| 341 }; | |
| 342 | 234 |
| 343 } // namespace | 235 } // namespace |
| 344 | 236 |
| 345 MotionEventBuffer::MotionEventBuffer(MotionEventBufferClient* client, | 237 MotionEventBuffer::MotionEventBuffer(MotionEventBufferClient* client, |
| 346 bool enable_resampling) | 238 bool enable_resampling) |
| 347 : client_(client), resample_(enable_resampling) { | 239 : client_(client), resample_(enable_resampling) { |
| 348 } | 240 } |
| 349 | 241 |
| 350 MotionEventBuffer::~MotionEventBuffer() { | 242 MotionEventBuffer::~MotionEventBuffer() { |
| 351 } | 243 } |
| 352 | 244 |
| 353 void MotionEventBuffer::OnMotionEvent(const MotionEvent& event) { | 245 void MotionEventBuffer::OnMotionEvent(const MotionEvent& event) { |
| 246 DCHECK_EQ(0U, event.GetHistorySize()); |
| 354 if (event.GetAction() != MotionEvent::ACTION_MOVE) { | 247 if (event.GetAction() != MotionEvent::ACTION_MOVE) { |
| 355 last_extrapolated_event_time_ = base::TimeTicks(); | 248 last_extrapolated_event_time_ = base::TimeTicks(); |
| 356 if (!buffered_events_.empty()) | 249 if (!buffered_events_.empty()) |
| 357 FlushWithoutResampling(buffered_events_.Pass()); | 250 FlushWithoutResampling(buffered_events_.Pass()); |
| 358 client_->ForwardMotionEvent(event); | 251 client_->ForwardMotionEvent(event); |
| 359 return; | 252 return; |
| 360 } | 253 } |
| 361 | 254 |
| 362 // Guard against events that are *older* than the last one that may have been | 255 // Guard against events that are *older* than the last one that may have been |
| 363 // artificially synthesized. | 256 // artificially synthesized. |
| 364 if (!last_extrapolated_event_time_.is_null()) { | 257 if (!last_extrapolated_event_time_.is_null()) { |
| 365 DCHECK(buffered_events_.empty()); | 258 DCHECK(buffered_events_.empty()); |
| 366 if (event.GetEventTime() < last_extrapolated_event_time_) | 259 if (event.GetEventTime() < last_extrapolated_event_time_) |
| 367 return; | 260 return; |
| 368 last_extrapolated_event_time_ = base::TimeTicks(); | 261 last_extrapolated_event_time_ = base::TimeTicks(); |
| 369 } | 262 } |
| 370 | 263 |
| 371 scoped_ptr<MotionEvent> clone = event.Clone(); | 264 scoped_ptr<MotionEventGeneric> clone = MotionEventGeneric::CloneEvent(event); |
| 372 if (buffered_events_.empty()) { | 265 if (buffered_events_.empty()) { |
| 373 buffered_events_.push_back(clone.release()); | 266 buffered_events_.push_back(clone.release()); |
| 374 client_->SetNeedsFlush(); | 267 client_->SetNeedsFlush(); |
| 375 return; | 268 return; |
| 376 } | 269 } |
| 377 | 270 |
| 378 if (CanAddSample(*buffered_events_.front(), *clone)) { | 271 if (CanAddSample(*buffered_events_.front(), *clone)) { |
| 379 DCHECK(buffered_events_.back()->GetEventTime() <= clone->GetEventTime()); | 272 DCHECK(buffered_events_.back()->GetEventTime() <= clone->GetEventTime()); |
| 380 } else { | 273 } else { |
| 381 FlushWithoutResampling(buffered_events_.Pass()); | 274 FlushWithoutResampling(buffered_events_.Pass()); |
| (...skipping 22 matching lines...) Expand all Loading... |
| 404 return; | 297 return; |
| 405 } | 298 } |
| 406 | 299 |
| 407 if (!resample_ || (events.size() == 1 && buffered_events_.empty())) { | 300 if (!resample_ || (events.size() == 1 && buffered_events_.empty())) { |
| 408 FlushWithoutResampling(events.Pass()); | 301 FlushWithoutResampling(events.Pass()); |
| 409 if (!buffered_events_.empty()) | 302 if (!buffered_events_.empty()) |
| 410 client_->SetNeedsFlush(); | 303 client_->SetNeedsFlush(); |
| 411 return; | 304 return; |
| 412 } | 305 } |
| 413 | 306 |
| 414 CompoundMotionEvent resampled_event(events.Pass()); | 307 FlushWithResampling(events.Pass(), frame_time); |
| 415 base::TimeTicks original_event_time = resampled_event.GetEventTime(); | 308 } |
| 309 |
| 310 void MotionEventBuffer::FlushWithResampling(MotionEventVector events, |
| 311 base::TimeTicks resample_time) { |
| 312 DCHECK(!events.empty()); |
| 313 base::TimeTicks original_event_time = events.back()->GetEventTime(); |
| 416 const MotionEvent* next_event = | 314 const MotionEvent* next_event = |
| 417 !buffered_events_.empty() ? buffered_events_.front() : NULL; | 315 !buffered_events_.empty() ? buffered_events_.front() : nullptr; |
| 418 | 316 |
| 419 // Try to interpolate/extrapolate a new event at |frame_time|. Note that | 317 scoped_ptr<MotionEventGeneric> resampled_event = |
| 420 // |new_event|, if non-NULL, is owned by |resampled_event_|. | 318 ConsumeSamplesAndTryResampling(resample_time, events.Pass(), next_event); |
| 421 const MotionEvent* new_event = | 319 DCHECK(resampled_event); |
| 422 resampled_event.TryResample(frame_time, next_event); | |
| 423 | 320 |
| 424 // Log the extrapolated event time, guarding against subsequently queued | 321 // Log the extrapolated event time, guarding against subsequently queued |
| 425 // events that might have an earlier timestamp. | 322 // events that might have an earlier timestamp. |
| 426 if (!next_event && new_event && | 323 if (!next_event && resampled_event->GetEventTime() > original_event_time) { |
| 427 new_event->GetEventTime() > original_event_time) { | 324 last_extrapolated_event_time_ = resampled_event->GetEventTime(); |
| 428 last_extrapolated_event_time_ = new_event->GetEventTime(); | |
| 429 } else { | 325 } else { |
| 430 last_extrapolated_event_time_ = base::TimeTicks(); | 326 last_extrapolated_event_time_ = base::TimeTicks(); |
| 431 } | 327 } |
| 432 | 328 |
| 433 client_->ForwardMotionEvent(resampled_event); | 329 client_->ForwardMotionEvent(*resampled_event); |
| 434 if (!buffered_events_.empty()) | 330 if (!buffered_events_.empty()) |
| 435 client_->SetNeedsFlush(); | 331 client_->SetNeedsFlush(); |
| 436 } | 332 } |
| 437 | 333 |
| 438 void MotionEventBuffer::FlushWithoutResampling(MotionEventVector events) { | 334 void MotionEventBuffer::FlushWithoutResampling(MotionEventVector events) { |
| 439 last_extrapolated_event_time_ = base::TimeTicks(); | 335 last_extrapolated_event_time_ = base::TimeTicks(); |
| 440 if (events.empty()) | 336 if (events.empty()) |
| 441 return; | 337 return; |
| 442 | 338 |
| 443 if (events.size() == 1) { | 339 client_->ForwardMotionEvent(*ConsumeSamples(events.Pass())); |
| 444 // Avoid CompoundEvent creation to prevent unnecessary allocations. | |
| 445 scoped_ptr<MotionEvent> event(events.front()); | |
| 446 events.weak_clear(); | |
| 447 client_->ForwardMotionEvent(*event); | |
| 448 return; | |
| 449 } | |
| 450 | |
| 451 CompoundMotionEvent compound_event(events.Pass()); | |
| 452 client_->ForwardMotionEvent(compound_event); | |
| 453 } | 340 } |
| 454 | 341 |
| 455 } // namespace ui | 342 } // namespace ui |
| OLD | NEW |