// Copyright (c) 2013 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. #include "chrome/browser/ui/gesture_prefs_observer_factory_aura.h" #include #include "base/bind.h" #include "base/bind_helpers.h" #include "base/compiler_specific.h" #include "base/prefs/pref_change_registrar.h" #include "base/prefs/pref_service.h" #include "chrome/browser/chrome_notification_types.h" #include "chrome/browser/profiles/incognito_helpers.h" #include "chrome/browser/profiles/profile.h" #include "chrome/common/pref_names.h" #include "components/keyed_service/content/browser_context_dependency_manager.h" #include "components/pref_registry/pref_registry_syncable.h" #include "content/public/browser/notification_observer.h" #include "content/public/browser/notification_service.h" #include "content/public/browser/overscroll_configuration.h" #include "content/public/common/renderer_preferences.h" #include "ui/events/gestures/gesture_configuration.h" using ui::GestureConfiguration; namespace { struct OverscrollPref { const char* pref_name; content::OverscrollConfig config; }; const std::vector& GetOverscrollPrefs() { CR_DEFINE_STATIC_LOCAL(std::vector, overscroll_prefs, ()); if (overscroll_prefs.empty()) { using namespace content; const OverscrollPref kOverscrollPrefs[] = { { prefs::kOverscrollHorizontalThresholdComplete, OVERSCROLL_CONFIG_HORIZ_THRESHOLD_COMPLETE }, { prefs::kOverscrollVerticalThresholdComplete, OVERSCROLL_CONFIG_VERT_THRESHOLD_COMPLETE }, { prefs::kOverscrollMinimumThresholdStart, OVERSCROLL_CONFIG_HORIZ_THRESHOLD_START_TOUCHSCREEN }, { prefs::kOverscrollMinimumThresholdStartTouchpad, OVERSCROLL_CONFIG_HORIZ_THRESHOLD_START_TOUCHPAD }, { prefs::kOverscrollVerticalThresholdStart, OVERSCROLL_CONFIG_VERT_THRESHOLD_START }, { prefs::kOverscrollHorizontalResistThreshold, OVERSCROLL_CONFIG_HORIZ_RESIST_AFTER }, { prefs::kOverscrollVerticalResistThreshold, OVERSCROLL_CONFIG_VERT_RESIST_AFTER }, }; overscroll_prefs.assign(kOverscrollPrefs, kOverscrollPrefs + arraysize(kOverscrollPrefs)); } return overscroll_prefs; } // This class manages gesture configuration preferences. class GesturePrefsObserver : public KeyedService { public: explicit GesturePrefsObserver(PrefService* prefs); virtual ~GesturePrefsObserver(); // KeyedService implementation. virtual void Shutdown() OVERRIDE; private: // Notification callback invoked when browser-side preferences // are updated and need to be pushed into ui::GesturePreferences. void Update(); // Notification callback invoked when the fling deacceleration // gesture preferences are changed from chrome://gesture. // Broadcasts the changes all renderers where they are used. void Notify(); // Notification helper to push overscroll preferences into // content. void UpdateOverscrollPrefs(); PrefChangeRegistrar registrar_; PrefService* prefs_; DISALLOW_COPY_AND_ASSIGN(GesturePrefsObserver); }; // The list of prefs we want to observe. // Note that this collection of settings should correspond to the settings used // in ui/events/gestures/gesture_configuration.h const char* kPrefsToObserve[] = { prefs::kFlingAccelerationCurveCoefficient0, prefs::kFlingAccelerationCurveCoefficient1, prefs::kFlingAccelerationCurveCoefficient2, prefs::kFlingAccelerationCurveCoefficient3, prefs::kFlingMaxCancelToDownTimeInMs, prefs::kFlingMaxTapGapTimeInMs, prefs::kTabScrubActivationDelayInMS, prefs::kMaxSeparationForGestureTouchesInPixels, prefs::kSemiLongPressTimeInSeconds, }; const char* kFlingTouchpadPrefs[] = { prefs::kFlingCurveTouchpadAlpha, prefs::kFlingCurveTouchpadBeta, prefs::kFlingCurveTouchpadGamma }; const char* kFlingTouchscreenPrefs[] = { prefs::kFlingCurveTouchscreenAlpha, prefs::kFlingCurveTouchscreenBeta, prefs::kFlingCurveTouchscreenGamma, }; GesturePrefsObserver::GesturePrefsObserver(PrefService* prefs) : prefs_(prefs) { registrar_.Init(prefs); registrar_.RemoveAll(); base::Closure callback = base::Bind(&GesturePrefsObserver::Update, base::Unretained(this)); base::Closure notify_callback = base::Bind(&GesturePrefsObserver::Notify, base::Unretained(this)); for (size_t i = 0; i < arraysize(kPrefsToObserve); ++i) registrar_.Add(kPrefsToObserve[i], callback); const std::vector& overscroll_prefs = GetOverscrollPrefs(); for (size_t i = 0; i < overscroll_prefs.size(); ++i) registrar_.Add(overscroll_prefs[i].pref_name, callback); for (size_t i = 0; i < arraysize(kFlingTouchpadPrefs); ++i) registrar_.Add(kFlingTouchpadPrefs[i], notify_callback); for (size_t i = 0; i < arraysize(kFlingTouchscreenPrefs); ++i) registrar_.Add(kFlingTouchscreenPrefs[i], notify_callback); Update(); } GesturePrefsObserver::~GesturePrefsObserver() {} void GesturePrefsObserver::Shutdown() { registrar_.RemoveAll(); } void GesturePrefsObserver::Update() { GestureConfiguration::set_fling_acceleration_curve_coefficients(0, prefs_->GetDouble(prefs::kFlingAccelerationCurveCoefficient0)); GestureConfiguration::set_fling_acceleration_curve_coefficients(1, prefs_->GetDouble(prefs::kFlingAccelerationCurveCoefficient1)); GestureConfiguration::set_fling_acceleration_curve_coefficients(2, prefs_->GetDouble(prefs::kFlingAccelerationCurveCoefficient2)); GestureConfiguration::set_fling_acceleration_curve_coefficients(3, prefs_->GetDouble(prefs::kFlingAccelerationCurveCoefficient3)); GestureConfiguration::set_fling_max_cancel_to_down_time_in_ms( prefs_->GetInteger(prefs::kFlingMaxCancelToDownTimeInMs)); GestureConfiguration::set_fling_max_tap_gap_time_in_ms( prefs_->GetInteger(prefs::kFlingMaxTapGapTimeInMs)); GestureConfiguration::set_tab_scrub_activation_delay_in_ms( prefs_->GetInteger(prefs::kTabScrubActivationDelayInMS)); GestureConfiguration::set_semi_long_press_time_in_seconds( prefs_->GetDouble( prefs::kSemiLongPressTimeInSeconds)); GestureConfiguration::set_max_separation_for_gesture_touches_in_pixels( prefs_->GetDouble( prefs::kMaxSeparationForGestureTouchesInPixels)); UpdateOverscrollPrefs(); } void GesturePrefsObserver::UpdateOverscrollPrefs() { const std::vector& overscroll_prefs = GetOverscrollPrefs(); for (size_t i = 0; i < overscroll_prefs.size(); ++i) { content::SetOverscrollConfig(overscroll_prefs[i].config, static_cast(prefs_->GetDouble(overscroll_prefs[i].pref_name))); } } void GesturePrefsObserver::Notify() { // Must do a notify to distribute the changes to all renderers. content::NotificationService* service = content::NotificationService::current(); service->Notify(chrome::NOTIFICATION_BROWSER_FLING_CURVE_PARAMETERS_CHANGED, content::Source(this), content::NotificationService::NoDetails()); } } // namespace // static GesturePrefsObserverFactoryAura* GesturePrefsObserverFactoryAura::GetInstance() { return Singleton::get(); } GesturePrefsObserverFactoryAura::GesturePrefsObserverFactoryAura() : BrowserContextKeyedServiceFactory( "GesturePrefsObserverAura", BrowserContextDependencyManager::GetInstance()) {} GesturePrefsObserverFactoryAura::~GesturePrefsObserverFactoryAura() {} KeyedService* GesturePrefsObserverFactoryAura::BuildServiceInstanceFor( content::BrowserContext* profile) const { return new GesturePrefsObserver(static_cast(profile)->GetPrefs()); } void GesturePrefsObserverFactoryAura::RegisterOverscrollPrefs( user_prefs::PrefRegistrySyncable* registry) { const std::vector& overscroll_prefs = GetOverscrollPrefs(); for (size_t i = 0; i < overscroll_prefs.size(); ++i) { registry->RegisterDoublePref( overscroll_prefs[i].pref_name, content::GetOverscrollConfig(overscroll_prefs[i].config), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); } } void GesturePrefsObserverFactoryAura::RegisterFlingCurveParameters( user_prefs::PrefRegistrySyncable* registry) { content::RendererPreferences def_prefs; for (size_t i = 0; i < arraysize(kFlingTouchpadPrefs); i++) registry->RegisterDoublePref( kFlingTouchpadPrefs[i], def_prefs.touchpad_fling_profile[i], user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); for (size_t i = 0; i < arraysize(kFlingTouchscreenPrefs); i++) registry->RegisterDoublePref( kFlingTouchscreenPrefs[i], def_prefs.touchscreen_fling_profile[i], user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); } void GesturePrefsObserverFactoryAura::RegisterProfilePrefs( user_prefs::PrefRegistrySyncable* registry) { registry->RegisterDoublePref( prefs::kFlingAccelerationCurveCoefficient0, GestureConfiguration::fling_acceleration_curve_coefficients(0), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterDoublePref( prefs::kFlingAccelerationCurveCoefficient1, GestureConfiguration::fling_acceleration_curve_coefficients(1), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterDoublePref( prefs::kFlingAccelerationCurveCoefficient2, GestureConfiguration::fling_acceleration_curve_coefficients(2), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterDoublePref( prefs::kFlingAccelerationCurveCoefficient3, GestureConfiguration::fling_acceleration_curve_coefficients(3), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterIntegerPref( prefs::kFlingMaxCancelToDownTimeInMs, GestureConfiguration::fling_max_cancel_to_down_time_in_ms(), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterIntegerPref( prefs::kFlingMaxTapGapTimeInMs, GestureConfiguration::fling_max_tap_gap_time_in_ms(), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterIntegerPref( prefs::kTabScrubActivationDelayInMS, GestureConfiguration::tab_scrub_activation_delay_in_ms(), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterDoublePref( prefs::kSemiLongPressTimeInSeconds, GestureConfiguration::semi_long_press_time_in_seconds(), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); registry->RegisterDoublePref( prefs::kMaxSeparationForGestureTouchesInPixels, GestureConfiguration::max_separation_for_gesture_touches_in_pixels(), user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); RegisterOverscrollPrefs(registry); RegisterFlingCurveParameters(registry); } bool GesturePrefsObserverFactoryAura::ServiceIsCreatedWithBrowserContext() const { // Create the observer as soon as the profile is created. return true; } content::BrowserContext* GesturePrefsObserverFactoryAura::GetBrowserContextToUse( content::BrowserContext* context) const { // Use same gesture preferences on incognito windows. return chrome::GetBrowserContextRedirectedInIncognito(context); } bool GesturePrefsObserverFactoryAura::ServiceIsNULLWhileTesting() const { // Some tests replace the PrefService of the TestingProfile after the // GesturePrefsObserver has been created, which makes Shutdown() // remove the registrar from a non-existent PrefService. return true; }