forked from organicmaps/organicmaps
Added filter to location manager
This commit is contained in:
parent
d7a5432a79
commit
8887c1f807
6 changed files with 65 additions and 22 deletions
|
@ -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>
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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)
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
|
|
|
@ -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);
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue