// Copyright 2014 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #import "base/test/ios/wait_util.h" #import #include "base/logging.h" #include "base/mac/scoped_nsobject.h" #include "base/message_loop/message_loop.h" #include "base/test/test_timeouts.h" #include "base/timer/elapsed_timer.h" namespace base { namespace test { namespace ios { TimeDelta TimeUntilCondition(ProceduralBlock action, ConditionBlock condition, MessageLoop* message_loop, TimeDelta timeout) { ElapsedTimer timer; if (action) action(); if (timeout == TimeDelta()) timeout = TestTimeouts::action_timeout(); const TimeDelta spin_delay(TimeDelta::FromMilliseconds(10)); while (timer.Elapsed() < timeout && (!condition || !condition())) { SpinRunLoopWithMaxDelay(spin_delay); if (message_loop) { message_loop->RunUntilIdle(); } } TimeDelta elapsed = timer.Elapsed(); // If DCHECK is ever hit, check if |action| is doing something that is // taking an unreasonably long time, or if |condition| does not come // true quickly enough. Increase |timeout| only if necessary. DCHECK(!condition || condition()); return elapsed; } void WaitUntilCondition(ConditionBlock condition, MessageLoop* message_loop, TimeDelta timeout) { TimeUntilCondition(nil, condition, message_loop, timeout); } void WaitUntilCondition(ConditionBlock condition) { WaitUntilCondition(condition, nullptr, TimeDelta()); } void SpinRunLoopWithMaxDelay(TimeDelta max_delay) { scoped_nsobject beforeDate( [[NSDate alloc] initWithTimeIntervalSinceNow:max_delay.InSecondsF()]); [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:beforeDate]; } } // namespace ios } // namespace test } // namespace base