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:
Alex Zolotarev 2011-05-03 16:03:58 +02:00 committed by Alex Zolotarev
parent f91561f65c
commit 1c052fd1e5
12 changed files with 366 additions and 37 deletions

View file

@ -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();
}

View file

@ -179,8 +179,7 @@ public:
@end
extern "C" location::LocationService & GetLocationService()
location::LocationService * CreateAppleLocationService()
{
static AppleLocationService ls;
return ls;
return new AppleLocationService();
}

View file

@ -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();

View 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;
}

View file

@ -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 \

View file

@ -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

View file

@ -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
View 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
View 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

View 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);
}

View 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();
}

View file

@ -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