forked from organicmaps/organicmaps
Introduced LocationManager with multiple location services support
Added WiFi location service for Mac @TODO filter location events from different sources
This commit is contained in:
parent
f91561f65c
commit
1c052fd1e5
12 changed files with 366 additions and 37 deletions
|
@ -321,9 +321,9 @@ void FrameWork<TModel>::AddRedrawCommandSure()
|
|||
m_renderQueue.AddWindowHandle(m_windowHandle);
|
||||
|
||||
// initialize gps and compass subsystem
|
||||
GetLocationService().SetGpsObserver(
|
||||
GetLocationManager().SetGpsObserver(
|
||||
boost::bind(&this_type::OnGpsUpdate, this, _1));
|
||||
GetLocationService().SetCompassObserver(
|
||||
GetLocationManager().SetCompassObserver(
|
||||
boost::bind(&this_type::OnCompassUpdate, this, _1));
|
||||
}
|
||||
|
||||
|
@ -496,7 +496,7 @@ void FrameWork<TModel>::AddRedrawCommandSure()
|
|||
m_locationObserver = observer;
|
||||
m_centeringMode = ECenterAndScale;
|
||||
// by default, we always start in accurate mode
|
||||
GetLocationService().StartUpdate(true);
|
||||
GetLocationManager().StartUpdate(true);
|
||||
}
|
||||
|
||||
template <typename TModel>
|
||||
|
@ -505,7 +505,7 @@ void FrameWork<TModel>::AddRedrawCommandSure()
|
|||
// reset callback
|
||||
m_locationObserver.clear();
|
||||
m_centeringMode = EDoNothing;
|
||||
GetLocationService().StopUpdate();
|
||||
GetLocationManager().StopUpdate();
|
||||
m_locationState.TurnOff();
|
||||
Invalidate();
|
||||
}
|
||||
|
|
|
@ -179,8 +179,7 @@ public:
|
|||
|
||||
@end
|
||||
|
||||
extern "C" location::LocationService & GetLocationService()
|
||||
location::LocationService * CreateAppleLocationService()
|
||||
{
|
||||
static AppleLocationService ls;
|
||||
return ls;
|
||||
return new AppleLocationService();
|
||||
}
|
||||
|
|
|
@ -67,6 +67,8 @@ namespace location
|
|||
}
|
||||
|
||||
public:
|
||||
virtual ~LocationService() {}
|
||||
|
||||
void SetGpsObserver(TGpsCallback observer)
|
||||
{
|
||||
m_gpsObserver = observer;
|
||||
|
@ -85,4 +87,6 @@ namespace location
|
|||
|
||||
}
|
||||
|
||||
extern "C" location::LocationService & GetLocationService();
|
||||
extern "C" location::LocationService & GetLocationManager();
|
||||
extern "C" location::LocationService * CreateAppleLocationService();
|
||||
extern "C" location::LocationService * CreateWiFiLocationService();
|
||||
|
|
75
platform/location_manager.cpp
Normal file
75
platform/location_manager.cpp
Normal file
|
@ -0,0 +1,75 @@
|
|||
#include "location.hpp"
|
||||
|
||||
#include "../std/target_os.hpp"
|
||||
#include "../std/vector.hpp"
|
||||
#include "../std/bind.hpp"
|
||||
|
||||
/// Chooses most accurate data from different position services
|
||||
class PositionFilter
|
||||
{
|
||||
public:
|
||||
GpsInfo const & MostNewAndAccuratePosition()
|
||||
{
|
||||
}
|
||||
};
|
||||
|
||||
namespace location
|
||||
{
|
||||
class LocationManager : public LocationService
|
||||
{
|
||||
vector<LocationService *> m_services;
|
||||
|
||||
void OnGpsUpdate(GpsInfo const & info)
|
||||
{
|
||||
NotifyGpsObserver(info);
|
||||
}
|
||||
|
||||
void OnCompassUpdate(CompassInfo const & info)
|
||||
{
|
||||
NotifyCompassObserver(info);
|
||||
}
|
||||
|
||||
public:
|
||||
LocationManager()
|
||||
{
|
||||
LocationService * service;
|
||||
|
||||
#if defined(OMIM_OS_IPHONE) || defined(OMIM_OS_MAC)
|
||||
service = CreateAppleLocationService();
|
||||
service->SetGpsObserver(bind(&LocationManager::OnGpsUpdate, this, _1));
|
||||
service->SetCompassObserver(bind(&LocationManager::OnCompassUpdate, this, _1));
|
||||
m_services.push_back(service);
|
||||
#endif
|
||||
|
||||
#if defined(OMIM_OS_WINDOWS) || defined(OMIM_OS_MAC)
|
||||
service = CreateWiFiLocationService();
|
||||
service->SetGpsObserver(bind(&LocationManager::OnGpsUpdate, this, _1));
|
||||
m_services.push_back(service);
|
||||
#endif
|
||||
}
|
||||
|
||||
virtual ~LocationManager()
|
||||
{
|
||||
for (size_t i = 0; i < m_services.size(); ++i)
|
||||
delete m_services[i];
|
||||
}
|
||||
|
||||
virtual void StartUpdate(bool useAccurateMode)
|
||||
{
|
||||
for (size_t i = 0; i < m_services.size(); ++i)
|
||||
m_services[i]->StartUpdate(useAccurateMode);
|
||||
}
|
||||
|
||||
virtual void StopUpdate()
|
||||
{
|
||||
for (size_t i = 0; i < m_services.size(); ++i)
|
||||
m_services[i]->StopUpdate();
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
location::LocationService & GetLocationManager()
|
||||
{
|
||||
static location::LocationManager mgr;
|
||||
return mgr;
|
||||
}
|
|
@ -5,17 +5,20 @@ TEMPLATE = lib
|
|||
CONFIG += staticlib
|
||||
|
||||
ROOT_DIR = ..
|
||||
DEPENDENCIES = coding base
|
||||
DEPENDENCIES = coding base jansson
|
||||
|
||||
include($$ROOT_DIR/common.pri)
|
||||
|
||||
QT *= core network
|
||||
|
||||
!iphone* {
|
||||
INCLUDEPATH += $$ROOT_DIR/3party/jansson/src
|
||||
|
||||
SOURCES += \
|
||||
qtplatform.cpp \
|
||||
wifi_location_service.cpp \
|
||||
qt_download_manager.cpp \
|
||||
qt_download.cpp
|
||||
qt_download.cpp \
|
||||
|
||||
HEADERS += \
|
||||
qt_download_manager.hpp \
|
||||
|
@ -26,10 +29,23 @@ HEADERS += \
|
|||
platform.hpp \
|
||||
download_manager.hpp \
|
||||
location.hpp \
|
||||
wifi_info.hpp
|
||||
|
||||
macx|iphone* {
|
||||
OBJECTIVE_SOURCES += apple_location_service.mm
|
||||
LIBS += -framework CoreLocation -framework Foundation
|
||||
} else {
|
||||
SOURCES += qt_location_service.cpp
|
||||
SOURCES +=
|
||||
}
|
||||
|
||||
macx:!iphone* {
|
||||
OBJECTIVE_SOURCES += wifi_info_mac.mm
|
||||
LIBS += -framework CoreWLAN
|
||||
}
|
||||
win32 {
|
||||
SOURCES += wifi_info_windows.cpp
|
||||
}
|
||||
|
||||
# common sources for all platforms
|
||||
SOURCES += \
|
||||
location_manager.cpp \
|
||||
|
|
|
@ -8,7 +8,7 @@ DEPENDENCIES = platform coding base tomcrypt jansson
|
|||
|
||||
include($$ROOT_DIR/common.pri)
|
||||
|
||||
INCLUDEPATH += $$ROOT_DIR/3party/jansson $$ROOT_DIR/3party/jansson/src
|
||||
INCLUDEPATH += $$ROOT_DIR/3party/jansson/src
|
||||
|
||||
QT *= core network
|
||||
|
||||
|
|
|
@ -1,23 +0,0 @@
|
|||
#include "location.hpp"
|
||||
|
||||
using namespace location;
|
||||
|
||||
class QtLocationService : public LocationService
|
||||
{
|
||||
public:
|
||||
virtual void StartUpdate(bool /*useAccurateMode*/)
|
||||
{
|
||||
// @TODO
|
||||
}
|
||||
|
||||
virtual void StopUpdate()
|
||||
{
|
||||
// @TODO
|
||||
}
|
||||
};
|
||||
|
||||
extern "C" location::LocationService & GetLocationService()
|
||||
{
|
||||
static QtLocationService ls;
|
||||
return ls;
|
||||
}
|
26
platform/wifi_info.hpp
Normal file
26
platform/wifi_info.hpp
Normal file
|
@ -0,0 +1,26 @@
|
|||
#pragma once
|
||||
|
||||
#include "../std/function.hpp"
|
||||
|
||||
#include "../std/vector.hpp"
|
||||
#include "../std/string.hpp"
|
||||
|
||||
class WiFiInfo
|
||||
{
|
||||
class Impl;
|
||||
Impl * m_pImpl;
|
||||
|
||||
public:
|
||||
struct AccessPoint
|
||||
{
|
||||
string m_bssid; //!< for example, "33-12-03-5b-44-9a"
|
||||
string m_ssid; //!< name for APN
|
||||
string m_signalStrength; //!< for example, "-54"
|
||||
};
|
||||
|
||||
WiFiInfo();
|
||||
~WiFiInfo();
|
||||
|
||||
typedef boost::function<void (vector<WiFiInfo::AccessPoint> const &)> WifiRequestCallbackT;
|
||||
void RequestWiFiBSSIDs(WifiRequestCallbackT callback);
|
||||
};
|
101
platform/wifi_info_mac.mm
Normal file
101
platform/wifi_info_mac.mm
Normal file
|
@ -0,0 +1,101 @@
|
|||
#include "../std/target_os.hpp"
|
||||
#ifdef OMIM_OS_MAC
|
||||
|
||||
#include "wifi_info.hpp"
|
||||
|
||||
#include "../base/string_utils.hpp"
|
||||
|
||||
#import <Foundation/Foundation.h>
|
||||
#import <CoreWLAN/CWInterface.h>
|
||||
#import <CoreWLAN/CWNetwork.h>
|
||||
|
||||
static string AppendZeroIfNeeded(string const & macAddrPart)
|
||||
{
|
||||
string res(macAddrPart);
|
||||
if (res.size() == 1)
|
||||
res.insert(0, "0");
|
||||
return res;
|
||||
}
|
||||
|
||||
@interface WiFiInfoMac : NSObject {
|
||||
@private
|
||||
WiFiInfo::WifiRequestCallbackT m_callback;
|
||||
vector<WiFiInfo::AccessPoint> m_accessPoints;
|
||||
}
|
||||
@end
|
||||
//- (id)InitWithCallback:(WiFiInfo::WifiRequestCallbackT)callback;
|
||||
@implementation WiFiInfoMac
|
||||
|
||||
- (id)InitWithCallback:(WiFiInfo::WifiRequestCallbackT)callback
|
||||
{
|
||||
self = [super init];
|
||||
m_callback = callback;
|
||||
[self performSelectorInBackground:@selector(WifiScanThread) withObject:nil];
|
||||
return self;
|
||||
}
|
||||
|
||||
- (void)dealloc
|
||||
{
|
||||
[super dealloc];
|
||||
NSLog(@"Mac OS WiFiInfo selfdestructed successfully");
|
||||
}
|
||||
|
||||
/// Executed on main thread
|
||||
- (void)NotifyGUI
|
||||
{
|
||||
m_callback(m_accessPoints);
|
||||
// selfdestruct
|
||||
[self release];
|
||||
}
|
||||
|
||||
/// new background thread
|
||||
- (void)WifiScanThread
|
||||
{
|
||||
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
|
||||
|
||||
m_accessPoints.clear();
|
||||
|
||||
CWInterface * iFace = [CWInterface interface];
|
||||
NSArray * nets = [iFace scanForNetworksWithParameters:nil error:nil];
|
||||
for (NSUInteger i = 0; i < [nets count]; ++i)
|
||||
{
|
||||
CWNetwork * net = (CWNetwork *)[nets objectAtIndex:i];
|
||||
WiFiInfo::AccessPoint apn;
|
||||
apn.m_ssid = [net.ssid UTF8String];
|
||||
apn.m_signalStrength = utils::to_string([net.rssi intValue]);
|
||||
// fix formatting for wifi address
|
||||
string const rawBssid = [net.bssid UTF8String];
|
||||
if (!rawBssid.empty())
|
||||
{
|
||||
utils::TokenizeIterator tokIt(rawBssid, ":");
|
||||
apn.m_bssid = AppendZeroIfNeeded(*tokIt);
|
||||
while (!(++tokIt).is_last())
|
||||
{
|
||||
apn.m_bssid += "-";
|
||||
apn.m_bssid += AppendZeroIfNeeded(*tokIt);
|
||||
}
|
||||
}
|
||||
m_accessPoints.push_back(apn);
|
||||
}
|
||||
|
||||
[pool drain];
|
||||
|
||||
[self performSelectorOnMainThread:@selector(NotifyGUI) withObject:nil waitUntilDone:NO];
|
||||
}
|
||||
@end
|
||||
/////////////////////////////////////////////////////////////////////////////
|
||||
WiFiInfo::WiFiInfo()
|
||||
{
|
||||
}
|
||||
|
||||
WiFiInfo::~WiFiInfo()
|
||||
{
|
||||
}
|
||||
|
||||
void WiFiInfo::RequestWiFiBSSIDs(WifiRequestCallbackT callback)
|
||||
{
|
||||
// it will be self-destroyed when finished
|
||||
[[WiFiInfoMac alloc] InitWithCallback:callback];
|
||||
}
|
||||
|
||||
#endif
|
34
platform/wifi_info_windows.cpp
Normal file
34
platform/wifi_info_windows.cpp
Normal file
|
@ -0,0 +1,34 @@
|
|||
#include "wifi_info.hpp"
|
||||
|
||||
#if defined(OMIM_OS_WINDOWS)
|
||||
#include "internal/wifi_impl_windows.hpp"
|
||||
#elif defined(OMIM_OS_MAC)
|
||||
#include "internal/wifi_impl_mac.hpp"
|
||||
#else
|
||||
#error "Please add your OS implementation"
|
||||
#endif
|
||||
|
||||
class WiFiInfo::Impl
|
||||
{
|
||||
WiFiInfo::WifiRequestCallbackT m_callback;
|
||||
|
||||
public:
|
||||
void RequestWiFiBSSIDs(WifiRequestCallbackT callback)
|
||||
{
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
WiFiInfo::WiFiInfo() : m_pImpl(new WiFiInfo::Impl)
|
||||
{
|
||||
}
|
||||
|
||||
WiFiInfo::~WiFiInfo()
|
||||
{
|
||||
delete m_pImpl;
|
||||
}
|
||||
|
||||
void WiFiInfo::RequestWiFiBSSIDs(WifiRequestCallbackT callback)
|
||||
{
|
||||
m_pImpl->RequestWiFiBSSIDs(callback);
|
||||
}
|
96
platform/wifi_location_service.cpp
Normal file
96
platform/wifi_location_service.cpp
Normal file
|
@ -0,0 +1,96 @@
|
|||
#include "location.hpp"
|
||||
#include "wifi_info.hpp"
|
||||
#include "download_manager.hpp"
|
||||
|
||||
#include "../base/logging.hpp"
|
||||
|
||||
#include "../std/bind.hpp"
|
||||
|
||||
#include "../3party/jansson/myjansson.hpp"
|
||||
|
||||
#define MWM_GEOLOCATION_SERVER "http://geolocation.server/"
|
||||
|
||||
namespace location
|
||||
{
|
||||
class WiFiLocationService : public LocationService
|
||||
{
|
||||
WiFiInfo m_wifiInfo;
|
||||
|
||||
void OnHttpPostFinished(HttpFinishedParams const & result)
|
||||
{
|
||||
// here we should receive json reply with coordinates and accuracy
|
||||
try
|
||||
{
|
||||
my::Json root(result.m_data.c_str());
|
||||
if (json_is_object(root))
|
||||
{
|
||||
json_t * location = json_object_get(root, "location");
|
||||
if (json_is_object(location))
|
||||
{
|
||||
json_t * lat = json_object_get(location, "latitude");
|
||||
json_t * lon = json_object_get(location, "longitude");
|
||||
json_t * acc = json_object_get(location, "accuracy");
|
||||
if (json_is_real(lat) && json_is_real(lon) && json_is_real(acc))
|
||||
{
|
||||
GpsInfo info;
|
||||
info.m_latitude = json_real_value(lat);
|
||||
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;
|
||||
NotifyGpsObserver(info);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
catch (my::Json::Exception const & e)
|
||||
{
|
||||
LOG(LWARNING, ("Invalid reply from location server:", e.what(), result.m_data));
|
||||
}
|
||||
LOG(LWARNING, ("Invalid reply from location server:", result.m_data));
|
||||
}
|
||||
|
||||
void OnWifiScanCompleted(vector<WiFiInfo::AccessPoint> const & accessPoints)
|
||||
{
|
||||
string jsonRequest("{\"version\":\"1.1.0\"");
|
||||
if (accessPoints.size())
|
||||
jsonRequest += ",\"wifi_towers\":[";
|
||||
|
||||
for (size_t i = 0; i < accessPoints.size(); ++i)
|
||||
{
|
||||
jsonRequest += "{\"mac_address\":\"";
|
||||
jsonRequest += accessPoints[i].m_bssid;
|
||||
jsonRequest += "\",\"ssid\":\"";
|
||||
jsonRequest += accessPoints[i].m_ssid;
|
||||
jsonRequest += "\",\"signal_strength\":";
|
||||
jsonRequest += accessPoints[i].m_signalStrength;
|
||||
jsonRequest += "},";
|
||||
}
|
||||
if (accessPoints.size())
|
||||
jsonRequest[jsonRequest.size() - 1] = ']';
|
||||
jsonRequest += "}";
|
||||
|
||||
HttpStartParams params;
|
||||
params.m_finish = bind(&WiFiLocationService::OnHttpPostFinished, this, _1);
|
||||
params.m_contentType = "application/json";
|
||||
params.m_postData = jsonRequest;
|
||||
params.m_url = MWM_GEOLOCATION_SERVER;
|
||||
GetDownloadManager().HttpRequest(params);
|
||||
}
|
||||
|
||||
public:
|
||||
virtual void StartUpdate(bool)
|
||||
{
|
||||
m_wifiInfo.RequestWiFiBSSIDs(bind(&WiFiLocationService::OnWifiScanCompleted, this, _1));
|
||||
}
|
||||
|
||||
virtual void StopUpdate()
|
||||
{
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
location::LocationService * CreateWiFiLocationService()
|
||||
{
|
||||
return new location::WiFiLocationService();
|
||||
}
|
|
@ -1,6 +1,7 @@
|
|||
# Main application in qt.
|
||||
ROOT_DIR = ..
|
||||
DEPENDENCIES = words map storage indexer yg platform geometry coding base bzip2 freetype expat fribidi tomcrypt version
|
||||
DEPENDENCIES = words map storage indexer yg platform geometry coding base \
|
||||
bzip2 freetype expat fribidi tomcrypt jansson version
|
||||
|
||||
include($$ROOT_DIR/common.pri)
|
||||
|
||||
|
@ -15,7 +16,7 @@ win32 {
|
|||
}
|
||||
|
||||
macx {
|
||||
LIBS += -framework CoreLocation -framework Foundation
|
||||
LIBS += -framework CoreLocation -framework Foundation -framework CoreWLAN
|
||||
|
||||
ICON = res/mac.icns
|
||||
PLIST_FILE = Info.plist
|
||||
|
|
Loading…
Add table
Reference in a new issue