mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
231 lines
7.2 KiB
C++
231 lines
7.2 KiB
C++
/*
|
|
* Copyright (C) 2010 Apple Inc. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
|
|
* THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "platform/UserGestureIndicator.h"
|
|
|
|
#include "wtf/Assertions.h"
|
|
#include "wtf/CurrentTime.h"
|
|
#include "wtf/MainThread.h"
|
|
|
|
namespace blink {
|
|
|
|
namespace {
|
|
|
|
// User gestures timeout in 1 second.
|
|
const double userGestureTimeout = 1.0;
|
|
|
|
// For out of process tokens we allow a 10 second delay.
|
|
const double userGestureOutOfProcessTimeout = 10.0;
|
|
|
|
class GestureToken : public UserGestureToken {
|
|
public:
|
|
static PassRefPtr<UserGestureToken> create() { return adoptRef(new GestureToken); }
|
|
|
|
virtual ~GestureToken() { }
|
|
virtual bool hasGestures() const OVERRIDE
|
|
{
|
|
// Do not enforce timeouts for gestures which spawned javascript prompts.
|
|
if (m_consumableGestures < 1 || (WTF::currentTime() - m_timestamp > (m_outOfProcess ? userGestureOutOfProcessTimeout : userGestureTimeout) && !m_javascriptPrompt))
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
void addGesture()
|
|
{
|
|
m_consumableGestures++;
|
|
m_timestamp = WTF::currentTime();
|
|
}
|
|
|
|
void resetTimestamp()
|
|
{
|
|
m_timestamp = WTF::currentTime();
|
|
}
|
|
|
|
bool consumeGesture()
|
|
{
|
|
if (!m_consumableGestures)
|
|
return false;
|
|
m_consumableGestures--;
|
|
return true;
|
|
}
|
|
|
|
virtual void setOutOfProcess() OVERRIDE
|
|
{
|
|
if (WTF::currentTime() - m_timestamp > userGestureTimeout)
|
|
return;
|
|
if (hasGestures())
|
|
m_outOfProcess = true;
|
|
}
|
|
|
|
virtual void setJavascriptPrompt() OVERRIDE
|
|
{
|
|
if (WTF::currentTime() - m_timestamp > userGestureTimeout)
|
|
return;
|
|
if (hasGestures())
|
|
m_javascriptPrompt = true;
|
|
}
|
|
|
|
private:
|
|
GestureToken()
|
|
: m_consumableGestures(0)
|
|
, m_timestamp(0)
|
|
, m_outOfProcess(false)
|
|
, m_javascriptPrompt(false)
|
|
{
|
|
}
|
|
|
|
size_t m_consumableGestures;
|
|
double m_timestamp;
|
|
bool m_outOfProcess;
|
|
bool m_javascriptPrompt;
|
|
};
|
|
|
|
}
|
|
|
|
static bool isDefinite(ProcessingUserGestureState state)
|
|
{
|
|
return state == DefinitelyProcessingNewUserGesture || state == DefinitelyProcessingUserGesture || state == DefinitelyNotProcessingUserGesture;
|
|
}
|
|
|
|
ProcessingUserGestureState UserGestureIndicator::s_state = DefinitelyNotProcessingUserGesture;
|
|
UserGestureIndicator* UserGestureIndicator::s_topmostIndicator = 0;
|
|
bool UserGestureIndicator::s_processedUserGestureSinceLoad = false;
|
|
|
|
UserGestureIndicator::UserGestureIndicator(ProcessingUserGestureState state)
|
|
: m_previousState(s_state)
|
|
{
|
|
// Silently ignore UserGestureIndicators on non-main threads.
|
|
if (!isMainThread())
|
|
return;
|
|
|
|
// We overwrite s_state only if the caller is definite about the gesture state.
|
|
if (isDefinite(state)) {
|
|
if (!s_topmostIndicator) {
|
|
s_topmostIndicator = this;
|
|
m_token = GestureToken::create();
|
|
} else {
|
|
m_token = s_topmostIndicator->currentToken();
|
|
}
|
|
s_state = state;
|
|
}
|
|
|
|
if (state == DefinitelyProcessingNewUserGesture) {
|
|
static_cast<GestureToken*>(m_token.get())->addGesture();
|
|
s_processedUserGestureSinceLoad = true;
|
|
} else if (state == DefinitelyProcessingUserGesture && s_topmostIndicator == this) {
|
|
static_cast<GestureToken*>(m_token.get())->addGesture();
|
|
s_processedUserGestureSinceLoad = true;
|
|
}
|
|
ASSERT(isDefinite(s_state));
|
|
}
|
|
|
|
UserGestureIndicator::UserGestureIndicator(PassRefPtr<UserGestureToken> token)
|
|
: m_previousState(s_state)
|
|
{
|
|
// Silently ignore UserGestureIndicators on non-main threads.
|
|
if (!isMainThread())
|
|
return;
|
|
|
|
if (token) {
|
|
static_cast<GestureToken*>(token.get())->resetTimestamp();
|
|
if (!s_topmostIndicator) {
|
|
s_topmostIndicator = this;
|
|
m_token = token;
|
|
} else {
|
|
m_token = s_topmostIndicator->currentToken();
|
|
if (static_cast<GestureToken*>(token.get())->hasGestures()) {
|
|
static_cast<GestureToken*>(m_token.get())->addGesture();
|
|
static_cast<GestureToken*>(token.get())->consumeGesture();
|
|
}
|
|
}
|
|
s_state = DefinitelyProcessingUserGesture;
|
|
}
|
|
|
|
ASSERT(isDefinite(s_state));
|
|
}
|
|
|
|
UserGestureIndicator::~UserGestureIndicator()
|
|
{
|
|
if (!isMainThread())
|
|
return;
|
|
s_state = m_previousState;
|
|
if (s_topmostIndicator == this)
|
|
s_topmostIndicator = 0;
|
|
ASSERT(isDefinite(s_state));
|
|
}
|
|
|
|
bool UserGestureIndicator::processingUserGesture()
|
|
{
|
|
if (!isMainThread())
|
|
return false;
|
|
return s_topmostIndicator && static_cast<GestureToken*>(s_topmostIndicator->currentToken())->hasGestures() && (s_state == DefinitelyProcessingNewUserGesture || s_state == DefinitelyProcessingUserGesture);
|
|
}
|
|
|
|
bool UserGestureIndicator::consumeUserGesture()
|
|
{
|
|
if (!isMainThread() || !s_topmostIndicator)
|
|
return false;
|
|
return static_cast<GestureToken*>(s_topmostIndicator->currentToken())->consumeGesture();
|
|
}
|
|
|
|
UserGestureToken* UserGestureIndicator::currentToken()
|
|
{
|
|
if (!isMainThread() || !s_topmostIndicator)
|
|
return 0;
|
|
return s_topmostIndicator->m_token.get();
|
|
}
|
|
|
|
void UserGestureIndicator::clearProcessedUserGestureSinceLoad()
|
|
{
|
|
if (isMainThread())
|
|
s_processedUserGestureSinceLoad = false;
|
|
}
|
|
|
|
bool UserGestureIndicator::processedUserGestureSinceLoad()
|
|
{
|
|
if (!isMainThread())
|
|
return false;
|
|
return s_processedUserGestureSinceLoad;
|
|
}
|
|
|
|
UserGestureIndicatorDisabler::UserGestureIndicatorDisabler()
|
|
: m_savedState(UserGestureIndicator::s_state)
|
|
, m_savedIndicator(UserGestureIndicator::s_topmostIndicator)
|
|
{
|
|
RELEASE_ASSERT(isMainThread());
|
|
UserGestureIndicator::s_state = DefinitelyNotProcessingUserGesture;
|
|
UserGestureIndicator::s_topmostIndicator = 0;
|
|
}
|
|
|
|
UserGestureIndicatorDisabler::~UserGestureIndicatorDisabler()
|
|
{
|
|
RELEASE_ASSERT(isMainThread());
|
|
UserGestureIndicator::s_state = m_savedState;
|
|
UserGestureIndicator::s_topmostIndicator = m_savedIndicator;
|
|
}
|
|
|
|
}
|