diff --git a/android/jni/com/mapswithme/maps/Framework.cpp b/android/jni/com/mapswithme/maps/Framework.cpp index ebd056f214..5ed00edef9 100644 --- a/android/jni/com/mapswithme/maps/Framework.cpp +++ b/android/jni/com/mapswithme/maps/Framework.cpp @@ -54,9 +54,12 @@ namespace android g_framework = this; m_videoTimer = new VideoTimer(bind(&Framework::CallRepaint, this)); + + /* size_t const measurementsCount = 5; m_sensors[0].SetCount(measurementsCount); m_sensors[1].SetCount(measurementsCount); + */ } Framework::~Framework() diff --git a/android/jni/com/mapswithme/maps/Framework.hpp b/android/jni/com/mapswithme/maps/Framework.hpp index 2de5f7cc79..d2556c7127 100644 --- a/android/jni/com/mapswithme/maps/Framework.hpp +++ b/android/jni/com/mapswithme/maps/Framework.hpp @@ -46,7 +46,7 @@ namespace android double m_lastY1; //@} - math::AvgVector m_sensors[2]; + math::LowPassVector m_sensors[2]; scoped_ptr m_scheduledTask; bool m_wasLongClick; diff --git a/geometry/avg_vector.hpp b/geometry/avg_vector.hpp index 3750b4e8b1..5916fbfb30 100644 --- a/geometry/avg_vector.hpp +++ b/geometry/avg_vector.hpp @@ -77,4 +77,35 @@ namespace math CalcAverage(arr); } }; + + // Compass smoothing parameters + // We're using technique described in + // http://windowsteamblog.com/windows_phone/b/wpdev/archive/2010/09/08/using-the-accelerometer-on-windows-phone-7.aspx + // In short it's a combination of low-pass filter to smooth the + // small orientation changes and a threshold filter to get big changes fast. + // k in the following formula: O(n) = O(n-1) + k * (I - O(n - 1)); + // smoothed heading angle. doesn't always correspond to the m_headingAngle + // as we change the heading angle only if the delta between + // smoothedHeadingRad and new heading value is bigger than smoothingThreshold. + template class LowPassVector + { + typedef array ArrayT; + ArrayT m_val; + T m_factor; + + public: + LowPassVector() : m_factor(0.5) {} + void SetFactor(T t) { m_factor = t; } + + /// @param[in] Next measurement. + /// @param[out] Average value. + void Next(T * arr) + { + for (size_t i = 0; i < Dim; ++i) + { + m_val[i] = m_val[i] + m_factor * (arr[i] - m_val[i]); + arr[i] = m_val[i]; + } + } + }; } diff --git a/map/compass_filter.cpp b/map/compass_filter.cpp index d9d1a9b295..ba22fe8442 100644 --- a/map/compass_filter.cpp +++ b/map/compass_filter.cpp @@ -1,66 +1,53 @@ #include "compass_filter.hpp" -#include "location_state.hpp" -#include "../geometry/angles.hpp" - -#include "../base/logging.hpp" +//#include "../geometry/angles.hpp" #include "../platform/location.hpp" -#define LOW_PASS_FACTOR 0.5 +/* +namespace +{ + double const LOW_PASS_FACTOR = 0.5; + double const TRASHOLD = my::DegToRad(15.0); +} +*/ CompassFilter::CompassFilter() + : m_headingRad(0.0) { - m_headingRad = m_smoothedHeadingRad = 0; - - // Set hard smoothing treshold constant to 10 degrees. - // We can't assign it from location::CompassInfo::accuracy, because actually it's a - // declination between magnetic and true north in Android - // (and it may be very large in particular places on the Earth). - m_smoothingThreshold = my::DegToRad(10); } void CompassFilter::OnCompassUpdate(location::CompassInfo const & info) { - double const newHeadingRad = ((info.m_trueHeading >= 0.0) ? info.m_trueHeading : info.m_magneticHeading); + double const heading = ((info.m_trueHeading >= 0.0) ? info.m_trueHeading : info.m_magneticHeading); -#ifdef OMIM_OS_IPHONE +//#ifdef OMIM_OS_IPHONE // On iOS we shouldn't smooth the compass values. - m_headingRad = newHeadingRad; + m_headingRad = heading; +/* #else - double const newHeadingDelta = fabs(newHeadingRad - m_headingRad); - // if new heading lies outside the twice treshold radius we immediately accept it - if (newHeadingDelta > 2.0 * m_smoothingThreshold) + if (fabs(ang::GetShortestDistance(m_headingRad, heading)) >= TRASHOLD) { - m_headingRad = newHeadingRad; - m_smoothedHeadingRad = newHeadingRad; + m_headingRad = heading; } else { // else we smooth the received value with the following formula // O(n) = O(n-1) + k * (I - O(n - 1)); - m_smoothedHeadingRad = m_smoothedHeadingRad + LOW_PASS_FACTOR * (newHeadingRad - m_smoothedHeadingRad); - - // if the change is too small we won't change the compass value - if (newHeadingDelta > m_smoothingThreshold) - m_headingRad = m_smoothedHeadingRad; + m_headingRad = m_headingRad + LOW_PASS_FACTOR * (heading - m_headingRad); } #endif +*/ } double CompassFilter::GetHeadingRad() const { return m_headingRad; } - -double CompassFilter::GetHeadingHalfErrorRad() const -{ - return m_smoothingThreshold; -} diff --git a/map/compass_filter.hpp b/map/compass_filter.hpp index 78bd2ac502..8a24c4271f 100644 --- a/map/compass_filter.hpp +++ b/map/compass_filter.hpp @@ -1,38 +1,16 @@ #pragma once -namespace location -{ - class CompassInfo; -} +namespace location { class CompassInfo; } class CompassFilter { -private: double m_headingRad; - // Compass smoothing parameters - // We're using technique described in - // http://windowsteamblog.com/windows_phone/b/wpdev/archive/2010/09/08/using-the-accelerometer-on-windows-phone-7.aspx - // In short it's a combination of low-pass filter to smooth the - // small orientation changes and a threshold filter to get big changes fast. - // @{ - // k in the following formula: O(n) = O(n-1) + k * (I - O(n - 1)); - //double m_lowPassKoeff; - // smoothed heading angle. doesn't always correspond to the m_headingAngle - // as we change the heading angle only if the delta between - // smoothedHeadingRad and new heading value is bigger than smoothingThreshold. - double m_smoothedHeadingRad; - double m_smoothingThreshold; - // @} - public: - - // Constructor CompassFilter(); + // Getting new compass value void OnCompassUpdate(location::CompassInfo const & info); // get heading angle in radians. double GetHeadingRad() const; - // get half of heading error in radians. - double GetHeadingHalfErrorRad() const; }; diff --git a/map/location_state.cpp b/map/location_state.cpp index 0d3cc454fd..0cc2dd6cc2 100644 --- a/map/location_state.cpp +++ b/map/location_state.cpp @@ -1,8 +1,6 @@ #include "location_state.hpp" #include "navigator.hpp" #include "framework.hpp" -#include "compass_filter.hpp" -#include "change_viewport_task.hpp" #include "move_screen_task.hpp" #include "../graphics/display_list.hpp" @@ -10,13 +8,11 @@ #include "../graphics/pen.hpp" #include "../anim/controller.hpp" -#include "../anim/angle_interpolation.hpp" +//#include "../anim/angle_interpolation.hpp" #include "../gui/controller.hpp" -#include "../platform/location.hpp" #include "../platform/platform.hpp" -#include "../platform/settings.hpp" #include "../geometry/rect2d.hpp" #include "../geometry/transformations.hpp" @@ -25,6 +21,7 @@ #include "../base/logging.hpp" + namespace location { double const State::s_cacheRadius = 500; @@ -433,6 +430,7 @@ namespace location void State::CheckCompassRotation() { +/* #ifndef OMIM_OS_IPHONE if (m_headingInterpolation) @@ -479,8 +477,9 @@ namespace location m_headingInterpolation->Unlock(); #else +*/ m_drawHeading = m_compassFilter.GetHeadingRad(); -#endif +//#endif } void State::CheckCompassFollowing() diff --git a/map/location_state.hpp b/map/location_state.hpp index 77ed7d73b2..1d2c583d2e 100644 --- a/map/location_state.hpp +++ b/map/location_state.hpp @@ -2,6 +2,8 @@ #include "compass_filter.hpp" +#include "../gui/element.hpp" + #include "../platform/location.hpp" #include "../geometry/point2d.hpp" @@ -11,19 +13,12 @@ #include "../std/scoped_ptr.hpp" #include "../std/map.hpp" -#include "../gui/element.hpp" class Framework; -namespace anim -{ - class AngleInterpolation; -} +//namespace anim { class AngleInterpolation; } -namespace graphics -{ - class DisplayList; -} +namespace graphics { class DisplayList; } namespace location { @@ -71,8 +66,6 @@ namespace location ELocationProcessMode m_locationProcessMode; ECompassProcessMode m_compassProcessMode; - void FollowCompass(); - typedef gui::Element base_t; graphics::Color m_locationAreaColor; @@ -108,7 +101,7 @@ namespace location mutable vector m_boundRects; m2::RectD m_boundRect; - shared_ptr m_headingInterpolation; + //shared_ptr m_headingInterpolation; typedef map TCompassStatusListeners; TCompassStatusListeners m_compassStatusListeners; @@ -125,8 +118,11 @@ namespace location m2::PointD const & globalPt1, ScreenBase const & s); - public: + void CheckCompassRotation(); + void CheckCompassFollowing(); + void FollowCompass(); + public: struct Params : base_t::Params { graphics::Color m_locationAreaColor; @@ -172,9 +168,6 @@ namespace location void AnimateToPositionAndEnqueueFollowing(); void AnimateToPositionAndEnqueueLocationProcessMode(location::ELocationProcessMode mode); - void CheckCompassRotation(); - void CheckCompassFollowing(); - /// @name GPS location updates routine. //@{ void OnLocationUpdate(location::GpsInfo const & info);