Added filter to location manager

This commit is contained in:
Alex Zolotarev 2011-05-06 22:41:16 +03:00 committed by Alex Zolotarev
parent d7a5432a79
commit 8887c1f807
6 changed files with 65 additions and 22 deletions

View file

@ -239,22 +239,19 @@ void FrameWork<TModel>::AddRedrawCommandSure()
template <typename TModel>
void FrameWork<TModel>::OnGpsUpdate(location::GpsInfo const & info)
{
if (info.m_timestamp < location::POSITION_TIMEOUT_SECONDS)
{
// notify GUI that we received gps position
if (!(m_locationState & location::State::EGps) && m_locationObserver)
m_locationObserver();
// notify GUI that we received gps position
if (!(m_locationState & location::State::EGps) && m_locationObserver)
m_locationObserver();
m_locationState.UpdateGps(info);
if (m_centeringMode == ECenterAndScale)
{
CenterAndScaleViewport();
m_centeringMode = ECenterOnly;
}
else if (m_centeringMode == ECenterOnly)
CenterViewport(m_locationState.Position());
UpdateNow();
m_locationState.UpdateGps(info);
if (m_centeringMode == ECenterAndScale)
{
CenterAndScaleViewport();
m_centeringMode = ECenterOnly;
}
else if (m_centeringMode == ECenterOnly)
CenterViewport(m_locationState.Position());
UpdateNow();
}
template <typename TModel>

View file

@ -137,7 +137,8 @@ public:
info.m_verticalAccuracy = location.verticalAccuracy;
info.m_latitude = location.coordinate.latitude;
info.m_longitude = location.coordinate.longitude;
info.m_timestamp = -[location.timestamp timeIntervalSinceNow];
info.m_timestamp = [location.timestamp timeIntervalSince1970];
info.m_source = location::EAppleNative;
}
- (void)locationManager:(CLLocationManager *)manager
@ -160,7 +161,7 @@ public:
newInfo.m_x = newHeading.x;
newInfo.m_y = newHeading.y;
newInfo.m_z = newHeading.z;
newInfo.m_timestamp = -[newHeading.timestamp timeIntervalSinceNow];
newInfo.m_timestamp = [location.timestamp timeIntervalSince1970];
m_service->OnHeadingUpdate(newInfo);
}
#endif
@ -173,6 +174,8 @@ public:
{
GpsInfo info;
info.m_status = EDisabledByUser;
info.m_source = location::EAppleNative;
info.m_timestamp = [[NSDate date] timeIntervalSince1970];
m_service->OnLocationUpdate(info);
}
}

View file

@ -17,13 +17,20 @@ namespace location
EAccurateMode,
ERoughMode //!< in this mode compass is turned off
};
enum TLocationSource
{
EAppleNative,
EWindowsNative,
EGoogle
};
/// @note always check m_status before using this structure
class GpsInfo
{
public:
TLocationSource m_source;
TLocationStatus m_status;
double m_timestamp; //!< how many seconds ago the position was retrieved
double m_timestamp; //!< seconds from 1st Jan 1970
double m_latitude; //!< degrees
double m_longitude; //!< degrees
double m_horizontalAccuracy; //!< metres

View file

@ -4,12 +4,43 @@
#include "../std/vector.hpp"
#include "../std/bind.hpp"
double ApproxDistanceSquareInMetres(double lat1, double lon1, double lat2, double lon2)
{
double const m1 = (lat1 - lat2) / 111111.;
double const m2 = (lon1 - lon2) / 111111.;
return m1 * m1 + m2 * m2;
}
/// Chooses most accurate data from different position services
class PositionFilter
{
location::GpsInfo * m_prevLocation;
public:
location::GpsInfo const & MostNewAndAccuratePosition()
PositionFilter() : m_prevLocation(NULL) {}
~PositionFilter() { delete m_prevLocation; }
/// @return true if location should be sent to observers
bool Passes(location::GpsInfo const & newLocation)
{
if (time(NULL) - newLocation.m_timestamp > location::POSITION_TIMEOUT_SECONDS)
return false;
bool passes = true;
if (m_prevLocation)
{
if (newLocation.m_timestamp < m_prevLocation->m_timestamp)
passes = false;
else if (newLocation.m_source != m_prevLocation->m_source
&& newLocation.m_horizontalAccuracy > m_prevLocation->m_horizontalAccuracy
&& ApproxDistanceSquareInMetres(newLocation.m_latitude,
newLocation.m_longitude,
m_prevLocation->m_latitude,
m_prevLocation->m_longitude)
> newLocation.m_horizontalAccuracy * newLocation.m_horizontalAccuracy)
passes = false;
}
else
m_prevLocation = new location::GpsInfo(newLocation);
return true;
}
};
@ -18,10 +49,12 @@ namespace location
class LocationManager : public LocationService
{
vector<LocationService *> m_services;
PositionFilter m_filter;
void OnGpsUpdate(GpsInfo const & info)
{
NotifyGpsObserver(info);
if (m_filter.Passes(info))
NotifyGpsObserver(info);
}
void OnCompassUpdate(CompassInfo const & info)

View file

@ -69,11 +69,12 @@ static string AppendZeroIfNeeded(string const & macAddrPart)
{
utils::TokenizeIterator tokIt(rawBssid, ":");
apn.m_bssid = AppendZeroIfNeeded(*tokIt);
while (!(++tokIt).is_last())
do
{
++tokIt;
apn.m_bssid += "-";
apn.m_bssid += AppendZeroIfNeeded(*tokIt);
}
} while (!tokIt.is_last());
}
m_accessPoints.push_back(apn);
}

View file

@ -37,7 +37,9 @@ namespace location
info.m_longitude = json_real_value(lon);
info.m_horizontalAccuracy = json_real_value(acc);
// @TODO introduce flags to mark valid values
info.m_status = ERoughMode;
info.m_status = EAccurateMode;
info.m_timestamp = time(NULL);
info.m_source = location::EGoogle;
NotifyGpsObserver(info);
}
}