This source file includes following definitions.
- m_isVibrating
- vibrate
- cancelVibration
- timerStartFired
- timerStopFired
- pageVisibilityChanged
- didCommitLoad
- vibrate
- vibrate
- from
- supplementName
#include "config.h"
#include "modules/vibration/NavigatorVibration.h"
#include "core/frame/LocalFrame.h"
#include "core/frame/Navigator.h"
#include "core/page/PageVisibilityState.h"
#include "public/platform/Platform.h"
#include "public/platform/WebVibration.h"
namespace WebCore {
const unsigned kVibrationPatternLengthMax = 99;
NavigatorVibration::NavigatorVibration(Page& page)
: PageLifecycleObserver(&page)
, m_timerStart(this, &NavigatorVibration::timerStartFired)
, m_timerStop(this, &NavigatorVibration::timerStopFired)
, m_isVibrating(false)
{
}
NavigatorVibration::~NavigatorVibration()
{
if (m_isVibrating)
cancelVibration();
}
bool NavigatorVibration::vibrate(const VibrationPattern& pattern)
{
VibrationPattern sanitized = pattern;
size_t length = sanitized.size();
if (length > kVibrationPatternLengthMax) {
sanitized.shrink(kVibrationPatternLengthMax);
length = kVibrationPatternLengthMax;
}
for (size_t i = 0; i < length; ++i) {
if (sanitized[i] > blink::kVibrationDurationMax)
sanitized[i] = blink::kVibrationDurationMax;
}
if (length && !(length % 2))
sanitized.removeLast();
if (m_isVibrating)
cancelVibration();
m_pattern = sanitized;
if (m_timerStart.isActive())
m_timerStart.stop();
if (!m_pattern.size())
return true;
if (m_pattern.size() == 1 && !m_pattern[0]) {
m_pattern.clear();
return true;
}
m_timerStart.startOneShot(0, FROM_HERE);
m_isVibrating = true;
return true;
}
void NavigatorVibration::cancelVibration()
{
m_pattern.clear();
if (m_isVibrating) {
blink::Platform::current()->cancelVibration();
m_isVibrating = false;
m_timerStop.stop();
}
}
void NavigatorVibration::timerStartFired(Timer<NavigatorVibration>* timer)
{
ASSERT_UNUSED(timer, timer == &m_timerStart);
if (m_pattern.size()) {
m_isVibrating = true;
blink::Platform::current()->vibrate(m_pattern[0]);
m_timerStop.startOneShot(m_pattern[0] / 1000.0, FROM_HERE);
m_pattern.remove(0);
}
}
void NavigatorVibration::timerStopFired(Timer<NavigatorVibration>* timer)
{
ASSERT_UNUSED(timer, timer == &m_timerStop);
if (m_pattern.isEmpty())
m_isVibrating = false;
if (m_pattern.size()) {
m_timerStart.startOneShot(m_pattern[0] / 1000.0, FROM_HERE);
m_pattern.remove(0);
}
}
void NavigatorVibration::pageVisibilityChanged()
{
if (page()->visibilityState() != PageVisibilityStateVisible)
cancelVibration();
}
void NavigatorVibration::didCommitLoad(LocalFrame* frame)
{
cancelVibration();
}
bool NavigatorVibration::vibrate(Navigator& navigator, unsigned time)
{
VibrationPattern pattern;
pattern.append(time);
return NavigatorVibration::vibrate(navigator, pattern);
}
bool NavigatorVibration::vibrate(Navigator& navigator, const VibrationPattern& pattern)
{
Page* page = navigator.frame()->page();
if (!page)
return false;
if (page->visibilityState() != PageVisibilityStateVisible)
return false;
return NavigatorVibration::from(*page).vibrate(pattern);
}
NavigatorVibration& NavigatorVibration::from(Page& page)
{
NavigatorVibration* navigatorVibration = static_cast<NavigatorVibration*>(Supplement<Page>::from(page, supplementName()));
if (!navigatorVibration) {
navigatorVibration = new NavigatorVibration(page);
Supplement<Page>::provideTo(page, supplementName(), adoptPtr(navigatorVibration));
}
return *navigatorVibration;
}
const char* NavigatorVibration::supplementName()
{
return "NavigatorVibration";
}
}