Refactored gps location system:

- Moved all code to crossplatform framework
- OS-dependent location services are implemented in platform
- Added my position button to desktops
This commit is contained in:
Alex Zolotarev 2011-04-23 22:46:43 +02:00 committed by Alex Zolotarev
parent f38bfeb80e
commit 1c20e772a0
26 changed files with 410 additions and 840 deletions

View file

@ -1,5 +1,4 @@
#import <UIKit/UIKit.h>
#import "UserLocationController.h"
#include "../../geometry/point2d.hpp"
#include "../../yg/texture.hpp"
@ -9,7 +8,6 @@
#include "../../map/navigator.hpp"
#include "../../map/feature_vec_model.hpp"
#include "../../std/shared_ptr.hpp"
#include "../../map/locator.hpp"
@interface MapViewController : UIViewController
{
@ -20,7 +18,6 @@
SCALING
} m_CurrentAction;
bool m_isDirtyPosition;
bool m_isSticking;
size_t m_StickyThreshold;
int m_iconSequenceNumber;
@ -37,14 +34,7 @@
- (void) ZoomToRect: (m2::RectD const &) rect;
- (void) UpdateIcon: (NSTimer *)theTimer;
- (void) OnUpdateLocation: (m2::PointD const &) mercatorPoint
withErrorRadius: (double) errorRadius
withLocTimeStamp: (double) locTimeStamp
withCurTimeStamp: (double) curTimeStamp;
- (void) OnChangeLocatorMode: (Locator::EMode) oldMode
withNewMode: (Locator::EMode) newMode;
//- (void) UpdateIcon: (NSTimer *)theTimer;
- (void) onResize: (GLint)width withHeight: (GLint)height;
- (void) onPaint;

View file

@ -4,7 +4,6 @@
#import "EAGLView.h"
#import "WindowHandle.h"
#import "../Settings/SettingsManager.h"
#import "IPhoneLocator.h"
#include "RenderContext.hpp"
#include "../../geometry/rect2d.hpp"
@ -23,7 +22,6 @@ typedef FrameWork<model::FeaturesFetcher, Navigator> framework_t;
// @TODO Make m_framework and m_storage MapsAppDelegate properties instead of global variables.
framework_t * m_framework = NULL;
shared_ptr<Locator> m_locator;
storage::Storage m_storage;
- (void) ZoomToRect: (m2::RectD const &) rect
@ -53,52 +51,28 @@ storage::Storage m_storage;
*/
}
- (void)OnLocationUpdated
{
m_myPositionButton.image = [UIImage imageNamed:@"location.png"];
}
- (IBAction)OnMyPositionClicked:(id)sender
{
if (m_locator->isRunning())
{
m_locator->stop();
m_framework->StopLocator();
((UIBarButtonItem*) sender).style = UIBarButtonItemStyleBordered;
}
else
{
m_framework->StartLocator(Locator::EPreciseMode);
m_locator->start(Locator::EPreciseMode);
m_isDirtyPosition = true;
((UIBarButtonItem*)sender).style = UIBarButtonItemStyleDone;
((UIBarButtonItem*)sender).image = [UIImage imageNamed:@"location-search.png"];
}
}
- (void) OnChangeLocatorMode:(Locator::EMode) oldMode
withNewMode:(Locator::EMode) newMode
{
if (newMode == Locator::ERoughMode)
if (((UIBarButtonItem *)sender).style == UIBarButtonItemStyleBordered)
{
m_myPositionButton.image = [UIImage imageNamed:@"location.png"];
// m_myPositionButton.style = UIBarButtonItemStyleBordered;
typedef void (*OnLocationUpdatedFunc)(id, SEL);
SEL onLocUpdatedSel = @selector(OnLocationUpdated);
OnLocationUpdatedFunc locUpdatedImpl = (OnLocationUpdatedFunc)[self methodForSelector:onLocUpdatedSel];
m_framework->StartLocationService(bind(locUpdatedImpl, self, onLocUpdatedSel));
((UIBarButtonItem *)sender).style = UIBarButtonItemStyleDone;
((UIBarButtonItem *)sender).image = [UIImage imageNamed:@"location-search.png"];
}
}
- (void) OnUpdateLocation: (m2::PointD const &) mercatorPoint
withErrorRadius: (double) errorRadius
withLocTimeStamp: (double) locTimeStamp
withCurTimeStamp: (double) curTimeStamp
{
if (m_isDirtyPosition && (curTimeStamp - locTimeStamp <= 10) && (m_locator->mode() == Locator::EPreciseMode))
else
{
/*
if (m_iconTimer != nil)
{
[m_iconTimer invalidate];
m_iconTimer = nil;
}
*/
m_framework->StopLocationService();
((UIBarButtonItem *)sender).style = UIBarButtonItemStyleBordered;
m_myPositionButton.image = [UIImage imageNamed:@"location.png"];
m_myPositionButton.style = UIBarButtonItemStyleDone;
m_isDirtyPosition = false;
}
}
@ -130,7 +104,6 @@ storage::Storage m_storage;
- (void) dealloc
{
m_locator.reset();
delete m_framework;
[super dealloc];
}
@ -145,28 +118,12 @@ storage::Storage m_storage;
shared_ptr<iphone::WindowHandle> windowHandle = [(EAGLView*)self.view windowHandle];
shared_ptr<yg::ResourceManager> resourceManager = [(EAGLView*)self.view resourceManager];
m_locator = shared_ptr<Locator>(new iphone::Locator());
// tricky boost::bind for objC class methods
typedef void (*TUpdateLocationFunc)(id, SEL, m2::PointD const &, double, double, double);
SEL updateLocationSel = @selector(OnUpdateLocation:withErrorRadius:withLocTimeStamp:withCurTimeStamp:);
TUpdateLocationFunc updateLocationImpl = (TUpdateLocationFunc)[self methodForSelector:updateLocationSel];
typedef void (*TChangeModeFunc)(id, SEL, Locator::EMode, Locator::EMode);
SEL changeModeSel = @selector(OnChangeLocatorMode:withNewMode:);
TChangeModeFunc changeModeImpl = (TChangeModeFunc)[self methodForSelector:changeModeSel];
m_locator->addOnUpdateLocationFn(boost::bind(updateLocationImpl, self, updateLocationSel, _1, _2, _3, _4));
m_locator->addOnChangeModeFn(boost::bind(changeModeImpl, self, changeModeSel, _1, _2));
m_framework = new framework_t(windowHandle, 40);
m_framework->InitStorage(m_storage);
m_framework->InitLocator(m_locator);
m_StickyThreshold = 10;
m_CurrentAction = NOTHING;
m_isDirtyPosition = false;
// initialize with currently active screen orientation
[self didRotateFromInterfaceOrientation: self.interfaceOrientation];

View file

@ -1,33 +0,0 @@
#import <Foundation/Foundation.h>
#import <CoreLocation/CoreLocation.h>
#include "../../../geometry/point2d.hpp"
@protocol UserLocationControllerDelegate
@required
- (void) OnLocation: (m2::PointD const &) mercatorPoint
withErrorRadius: (double) errorRadius
withTimestamp: (NSDate *) timestamp;
- (void) OnHeading: (CLHeading *)heading;
- (void) OnLocationError: (NSString *) errorDescription;
@end
@interface UserLocationController : NSObject<CLLocationManagerDelegate>
{
//@private
// CLLocationManager * m_locationManager;
@public
id delegate;
// BOOL active;
}
@property (nonatomic, assign) id delegate;
@property (nonatomic, assign) CLLocationManager * locationManager;
/// YES means that location manager is active
//@property (nonatomic, assign) BOOL active;
- (id) initWithDelegate: (id) locationDelegate;
@end

View file

@ -1,93 +0,0 @@
#import "UserLocationController.h"
#include "../../../indexer/mercator.hpp"
@implementation UserLocationController
@synthesize delegate;
//@synthesize active;
@synthesize locationManager;
- (id) init
{
self = [super init];
if (self != nil)
{
self.locationManager = [[CLLocationManager alloc] init];
self.locationManager.delegate = self;
// active = NO;
}
return self;
}
- (id) initWithDelegate: (id) locationDelegate
{
delegate = locationDelegate;
return [self init];
}
- (void) dealloc
{
// [self Stop];
[self.locationManager release];
[super dealloc];
}
/*- (void) Start
{
// active = YES;
[self.locationManager startUpdatingLocation];
self.locationManager.distanceFilter = kCLDistanceFilterNone;
if ([CLLocationManager headingAvailable])
{
self.locationManager.headingFilter = 1;
[self.locationManager startUpdatingHeading];
}
else
NSLog(@"heading information is not available");
}
- (void) Stop
{
[self.locationManager stopUpdatingLocation];
[self.locationManager stopUpdatingHeading];
// active = NO;
}*/
- (void) locationManager: (CLLocationManager *) manager
didUpdateHeading: (CLHeading *) newHeading
{
[self.delegate OnHeading: newHeading];
}
- (void) locationManager: (CLLocationManager *) manager
didUpdateToLocation: (CLLocation *) newLocation
fromLocation: (CLLocation *) oldLocation
{
// NSLog(@"NewLocation: %@", [newLocation description]);
m2::PointD mercPoint(MercatorBounds::LonToX(newLocation.coordinate.longitude),
MercatorBounds::LatToY(newLocation.coordinate.latitude));
m2::RectD errorRectXY = MercatorBounds::MetresToXY(newLocation.coordinate.longitude,
newLocation.coordinate.latitude,
newLocation.horizontalAccuracy);
double errorRadiusXY = sqrt((errorRectXY.SizeX() * errorRectXY.SizeX()
+ errorRectXY.SizeY() * errorRectXY.SizeY()) / 4);
[self.delegate OnLocation: mercPoint withErrorRadius: errorRadiusXY withTimestamp: newLocation.timestamp];
}
- (void) locationManager: (CLLocationManager *) manager
didFailWithError: (NSError *) error
{
[self.delegate OnLocationError: [error localizedDescription]];
if ([error code] == kCLErrorDenied)
{
// @TODO display some warning to the user about disabled location services,
// probably allow him to enable them by going to the settings
}
}
@end

View file

@ -1,57 +0,0 @@
//
// IPhoneLocator.h
// Maps
//
// Created by Siarhei Rachytski on 3/28/11.
// Copyright 2011 MapsWithMe. All rights reserved.
//
#include "../../map/locator.hpp"
#include "UserLocationController.h"
namespace iphone
{
class Locator;
}
@interface LocatorThunk : NSObject<UserLocationControllerDelegate>
{
iphone::Locator * m_locator;
}
@property (nonatomic, assign) iphone::Locator * locator;
- (void) initWithLocator : (iphone::Locator*) locator;
- (void) OnLocation: (m2::PointD const &) mercatorPoint
withErrorRadius: (double) errorRadius
withTimestamp: (NSDate *) timestamp;
- (void) OnHeading: (CLHeading *)heading;
- (void) OnLocationError: (NSString *) errorDescription;
@end
namespace iphone
{
class Locator : public ::Locator
{
private:
LocatorThunk * m_thunk;
UserLocationController * m_locationController;
public:
Locator();
~Locator();
void start(EMode mode);
void stop();
void setMode(EMode mode);
void locationUpdate(m2::PointD const & pt, double errorRadius, double locTimeStamp, double curTimeStamp);
void headingUpdate(double trueHeading, double magneticHeading, double accuracy);
};
}

View file

@ -1,112 +0,0 @@
//
// IPhoneLocator.mm
// Maps
//
// Created by Siarhei Rachytski on 3/28/11.
// Copyright 2011 MapsWithMe. All rights reserved.
//
#include "IPhoneLocator.h"
#import <Foundation/NSDate.h>
@implementation LocatorThunk
@synthesize locator;
- (void) initWithLocator : (iphone::Locator *) loc
{
self.locator = loc;
}
- (void) OnLocation: (m2::PointD const &) mercatorPoint
withErrorRadius: (double) errorRadius
withTimestamp: (NSDate *) timestamp;
{
self.locator->locationUpdate(mercatorPoint,
errorRadius,
[timestamp timeIntervalSince1970],
[[NSDate date] timeIntervalSince1970]);
}
- (void) OnHeading: (CLHeading*) heading
{
self.locator->headingUpdate(heading.trueHeading,
heading.magneticHeading,
heading.headingAccuracy);
}
- (void) OnLocationError: (NSString*) error
{
NSLog(@"Error : %@", error);
}
@end
namespace iphone
{
Locator::Locator()
{
m_thunk = [LocatorThunk alloc];
[m_thunk initWithLocator:this];
m_locationController = [[UserLocationController alloc] initWithDelegate:m_thunk];
}
Locator::~Locator()
{
stop();
[m_locationController release];
[m_thunk release];
}
void Locator::start(EMode mode)
{
::Locator::setMode(mode);
[m_locationController.locationManager startUpdatingLocation];
if (mode == ERoughMode)
m_locationController.locationManager.distanceFilter = 100;
else
m_locationController.locationManager.distanceFilter = kCLDistanceFilterNone;
if ([CLLocationManager headingAvailable])
{
m_locationController.locationManager.headingFilter = 1;
[m_locationController.locationManager startUpdatingHeading];
}
::Locator::start(mode);
}
void Locator::stop()
{
::Locator::stop();
[m_locationController.locationManager stopUpdatingLocation];
if ([CLLocationManager headingAvailable])
[m_locationController.locationManager stopUpdatingHeading];
}
void Locator::setMode(EMode mode)
{
EMode oldMode = ::Locator::mode();
::Locator::setMode(mode);
callOnChangeModeFns(oldMode, mode);
if (mode == ERoughMode)
m_locationController.locationManager.distanceFilter = 100;
else
m_locationController.locationManager.distanceFilter = kCLDistanceFilterNone;
}
void Locator::locationUpdate(m2::PointD const & pt, double errorRadius, double locTimeStamp, double curTimeStamp)
{
callOnUpdateLocationFns(pt, errorRadius, locTimeStamp, curTimeStamp);
}
void Locator::headingUpdate(double trueHeading, double magneticHeading, double accuracy)
{
callOnUpdateHeadingFns(trueHeading, magneticHeading, accuracy);
}
}

View file

@ -45,7 +45,6 @@
EE7F29811219ECA300EB67A9 /* RenderBuffer.mm in Sources */ = {isa = PBXBuildFile; fileRef = EE7F297D1219ECA300EB67A9 /* RenderBuffer.mm */; };
EE7F29821219ECA300EB67A9 /* RenderContext.mm in Sources */ = {isa = PBXBuildFile; fileRef = EE7F297E1219ECA300EB67A9 /* RenderContext.mm */; };
EE7F29831219ECA300EB67A9 /* WindowHandle.mm in Sources */ = {isa = PBXBuildFile; fileRef = EE7F29801219ECA300EB67A9 /* WindowHandle.mm */; };
EE9588BC134114F50056F2D5 /* IPhoneLocator.mm in Sources */ = {isa = PBXBuildFile; fileRef = EE9588BB134114F50056F2D5 /* IPhoneLocator.mm */; };
EEA61601134C496A003A9827 /* 01_dejavusans.ttf in Resources */ = {isa = PBXBuildFile; fileRef = EEA615E5134C4968003A9827 /* 01_dejavusans.ttf */; };
EEA61602134C496A003A9827 /* 02_wqy-microhei.ttf in Resources */ = {isa = PBXBuildFile; fileRef = EEA615E6134C4968003A9827 /* 02_wqy-microhei.ttf */; };
EEA61603134C496A003A9827 /* 03_jomolhari-id-a3d.ttf in Resources */ = {isa = PBXBuildFile; fileRef = EEA615E7134C4968003A9827 /* 03_jomolhari-id-a3d.ttf */; };
@ -54,7 +53,6 @@
EEAF65E5134BCBD500A81C82 /* location-search.png in Resources */ = {isa = PBXBuildFile; fileRef = EEAF65E3134BCBD500A81C82 /* location-search.png */; };
EEAF65E6134BCBD500A81C82 /* location-search@2x.png in Resources */ = {isa = PBXBuildFile; fileRef = EEAF65E4134BCBD500A81C82 /* location-search@2x.png */; };
EEB7E22211E9079400080A68 /* CoreLocation.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = EEB7E22111E9079400080A68 /* CoreLocation.framework */; };
EEB7E30911E9094F00080A68 /* UserLocationController.mm in Sources */ = {isa = PBXBuildFile; fileRef = EEB7E30811E9094F00080A68 /* UserLocationController.mm */; };
EED10A4511F78D120095FAD4 /* MapViewController.mm in Sources */ = {isa = PBXBuildFile; fileRef = EED10A4411F78D120095FAD4 /* MapViewController.mm */; };
EEE4C93E1298A2F3007231A9 /* symbols_24.png in Resources */ = {isa = PBXBuildFile; fileRef = EEE4C93D1298A2F3007231A9 /* symbols_24.png */; };
EEE4C9401298A303007231A9 /* symbols_48.png in Resources */ = {isa = PBXBuildFile; fileRef = EEE4C93F1298A303007231A9 /* symbols_48.png */; };
@ -69,6 +67,7 @@
FA0660001286167A00FEA989 /* Default.png in Resources */ = {isa = PBXBuildFile; fileRef = FA065FFE1286167A00FEA989 /* Default.png */; };
FA0660031286168700FEA989 /* Default-Portrait.png in Resources */ = {isa = PBXBuildFile; fileRef = FA0660011286168700FEA989 /* Default-Portrait.png */; };
FA0660041286168700FEA989 /* Default-Landscape.png in Resources */ = {isa = PBXBuildFile; fileRef = FA0660021286168700FEA989 /* Default-Landscape.png */; };
FA2EF9C713630C3B00E3E484 /* libplatform.a in Frameworks */ = {isa = PBXBuildFile; fileRef = FA2EF9C613630C3B00E3E484 /* libplatform.a */; };
FA34BECA1338D72F00FFB2A7 /* CustomAlertView.mm in Sources */ = {isa = PBXBuildFile; fileRef = FA34BEC81338D72F00FFB2A7 /* CustomAlertView.mm */; };
FA4135EA120A263C0062D5B4 /* CountriesViewController.mm in Sources */ = {isa = PBXBuildFile; fileRef = FA4135E2120A263C0062D5B4 /* CountriesViewController.mm */; };
FA4135ED120A263C0062D5B4 /* SettingsManager.mm in Sources */ = {isa = PBXBuildFile; fileRef = FA4135E7120A263C0062D5B4 /* SettingsManager.mm */; };
@ -145,7 +144,6 @@
EE7F297E1219ECA300EB67A9 /* RenderContext.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; lineEnding = 0; name = RenderContext.mm; path = Classes/RenderContext.mm; sourceTree = SOURCE_ROOT; };
EE7F297F1219ECA300EB67A9 /* WindowHandle.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; lineEnding = 0; name = WindowHandle.h; path = Classes/WindowHandle.h; sourceTree = SOURCE_ROOT; };
EE7F29801219ECA300EB67A9 /* WindowHandle.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; lineEnding = 0; name = WindowHandle.mm; path = Classes/WindowHandle.mm; sourceTree = SOURCE_ROOT; };
EE9588BB134114F50056F2D5 /* IPhoneLocator.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = IPhoneLocator.mm; sourceTree = "<group>"; };
EEA615E5134C4968003A9827 /* 01_dejavusans.ttf */ = {isa = PBXFileReference; lastKnownFileType = file; name = 01_dejavusans.ttf; path = ../../data/01_dejavusans.ttf; sourceTree = SOURCE_ROOT; };
EEA615E6134C4968003A9827 /* 02_wqy-microhei.ttf */ = {isa = PBXFileReference; lastKnownFileType = file; name = "02_wqy-microhei.ttf"; path = "../../data/02_wqy-microhei.ttf"; sourceTree = SOURCE_ROOT; };
EEA615E7134C4968003A9827 /* 03_jomolhari-id-a3d.ttf */ = {isa = PBXFileReference; lastKnownFileType = file; name = "03_jomolhari-id-a3d.ttf"; path = "../../data/03_jomolhari-id-a3d.ttf"; sourceTree = SOURCE_ROOT; };
@ -154,13 +152,10 @@
EEAF65E3134BCBD500A81C82 /* location-search.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "location-search.png"; sourceTree = "<group>"; };
EEAF65E4134BCBD500A81C82 /* location-search@2x.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "location-search@2x.png"; sourceTree = "<group>"; };
EEB7E22111E9079400080A68 /* CoreLocation.framework */ = {isa = PBXFileReference; lastKnownFileType = wrapper.framework; name = CoreLocation.framework; path = System/Library/Frameworks/CoreLocation.framework; sourceTree = SDKROOT; };
EEB7E30711E9094F00080A68 /* UserLocationController.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; lineEnding = 0; path = UserLocationController.h; sourceTree = "<group>"; xcLanguageSpecificationIdentifier = xcode.lang.cpp; };
EEB7E30811E9094F00080A68 /* UserLocationController.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; lineEnding = 0; path = UserLocationController.mm; sourceTree = "<group>"; xcLanguageSpecificationIdentifier = xcode.lang.objcpp; };
EED10A4411F78D120095FAD4 /* MapViewController.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; lineEnding = 0; path = MapViewController.mm; sourceTree = "<group>"; xcLanguageSpecificationIdentifier = xcode.lang.objcpp; };
EEE4C93D1298A2F3007231A9 /* symbols_24.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; name = symbols_24.png; path = ../../data/symbols_24.png; sourceTree = SOURCE_ROOT; };
EEE4C93F1298A303007231A9 /* symbols_48.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; name = symbols_48.png; path = ../../data/symbols_48.png; sourceTree = SOURCE_ROOT; };
EEE4C9411298A31B007231A9 /* basic_highres.skn */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; name = basic_highres.skn; path = ../../data/basic_highres.skn; sourceTree = SOURCE_ROOT; };
EEE8F9C2134106AB001DC91A /* IPhoneLocator.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = IPhoneLocator.h; sourceTree = "<group>"; };
EEF5745412DE1AD50082F472 /* libfribidi.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; path = libfribidi.a; sourceTree = SOURCE_ROOT; };
EEFE7C1212F8C9E1006AF8C3 /* fonts_blacklist.txt */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; name = fonts_blacklist.txt; path = ../../data/fonts_blacklist.txt; sourceTree = "<group>"; };
EEFE7C1312F8C9E1006AF8C3 /* fonts_whitelist.txt */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text; name = fonts_whitelist.txt; path = ../../data/fonts_whitelist.txt; sourceTree = "<group>"; };
@ -170,6 +165,7 @@
FA065FFE1286167A00FEA989 /* Default.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = Default.png; sourceTree = SOURCE_ROOT; };
FA0660011286168700FEA989 /* Default-Portrait.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "Default-Portrait.png"; sourceTree = SOURCE_ROOT; };
FA0660021286168700FEA989 /* Default-Landscape.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = "Default-Landscape.png"; sourceTree = SOURCE_ROOT; };
FA2EF9C613630C3B00E3E484 /* libplatform.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libplatform.a; path = libplatform.a; sourceTree = SOURCE_ROOT; };
FA34BEC81338D72F00FFB2A7 /* CustomAlertView.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; path = CustomAlertView.mm; sourceTree = "<group>"; };
FA34BEC91338D72F00FFB2A7 /* CustomAlertView.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = CustomAlertView.h; sourceTree = "<group>"; };
FA4135E1120A263C0062D5B4 /* CountriesViewController.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; lineEnding = 0; name = CountriesViewController.h; path = Settings/CountriesViewController.h; sourceTree = SOURCE_ROOT; xcLanguageSpecificationIdentifier = xcode.lang.objcpp; };
@ -224,6 +220,7 @@
FA8F8938132D5DB00048E3FE /* libtomcrypt.a in Frameworks */,
49DE1CA413437D7A00A93417 /* libbzip2.a in Frameworks */,
49DE1CA513437D7A00A93417 /* libwords.a in Frameworks */,
FA2EF9C713630C3B00E3E484 /* libplatform.a in Frameworks */,
);
runOnlyForDeploymentPostprocessing = 0;
};
@ -245,8 +242,6 @@
1D3623250D0F684500981E51 /* MapsAppDelegate.mm */,
46F8A2EB10EB63040045521A /* MapViewController.h */,
EED10A4411F78D120095FAD4 /* MapViewController.mm */,
EEB7E30711E9094F00080A68 /* UserLocationController.h */,
EEB7E30811E9094F00080A68 /* UserLocationController.mm */,
49DD2B4E132FA8880031D82E /* GuideViewController.h */,
49DD2B4F132FA8880031D82E /* GuideViewController.mm */,
);
@ -400,6 +395,7 @@
FA1DE68411E15D4E00C6D69A /* Static Libraries */ = {
isa = PBXGroup;
children = (
FA2EF9C613630C3B00E3E484 /* libplatform.a */,
49DE1CA213437D7A00A93417 /* libbzip2.a */,
49DE1CA313437D7A00A93417 /* libwords.a */,
FA8F8937132D5DB00048E3FE /* libtomcrypt.a */,
@ -449,8 +445,6 @@
FAF37EFA126DCE6F005EA154 /* IPhoneDownload.mm */,
FAF37EFE126DCE6F005EA154 /* IPhonePlatform.hpp */,
FAF37EFD126DCE6F005EA154 /* IPhonePlatform.mm */,
EEE8F9C2134106AB001DC91A /* IPhoneLocator.h */,
EE9588BB134114F50056F2D5 /* IPhoneLocator.mm */,
);
name = Platform;
sourceTree = "<group>";
@ -561,7 +555,6 @@
1D60589B0D05DD56006BFB54 /* main.mm in Sources */,
1D3623260D0F684500981E51 /* MapsAppDelegate.mm in Sources */,
46F26CD810F623BA00ECCA39 /* EAGLView.mm in Sources */,
EEB7E30911E9094F00080A68 /* UserLocationController.mm in Sources */,
EED10A4511F78D120095FAD4 /* MapViewController.mm in Sources */,
FA4135EA120A263C0062D5B4 /* CountriesViewController.mm in Sources */,
FA4135ED120A263C0062D5B4 /* SettingsManager.mm in Sources */,
@ -574,7 +567,6 @@
49DD2B51132FA8880031D82E /* GuideViewController.mm in Sources */,
FAFCB63613366E78001A5C59 /* WebViewController.mm in Sources */,
FA34BECA1338D72F00FFB2A7 /* CustomAlertView.mm in Sources */,
EE9588BC134114F50056F2D5 /* IPhoneLocator.mm in Sources */,
4938BB3B1343716500E0815A /* ArticleVC.mm in Sources */,
4938BB3C1343716500E0815A /* PerfCount.mm in Sources */,
4938BB3D1343716500E0815A /* SearchVC.mm in Sources */,

View file

@ -4,8 +4,8 @@
#include "drawer_yg.hpp"
#include "render_queue.hpp"
#include "information_display.hpp"
#include "locator.hpp"
#include "window_handle.hpp"
#include "location_state.hpp"
#include "../defines.hpp"
@ -16,6 +16,7 @@
#include "../indexer/feature.hpp"
#include "../platform/platform.hpp"
#include "../platform/location.hpp"
#include "../yg/defines.hpp"
#include "../yg/screen.hpp"
@ -40,8 +41,6 @@
#include "../std/target_os.hpp"
#include "../std/fstream.hpp"
#include "../base/start_mem_debug.hpp"
//#define DRAW_TOUCH_POINTS
namespace di { class DrawInfo; }
@ -86,6 +85,8 @@ namespace fwork
}
typedef boost::function<void (void)> LocationRetrievedCallbackT;
template
<
class TModel,
@ -101,7 +102,6 @@ class FrameWork
model_t m_model;
navigator_t m_navigator;
shared_ptr<WindowHandle> m_windowHandle;
shared_ptr<Locator> m_locator;
bool m_isBenchmarking;
bool m_isBenchmarkInitialized;
@ -118,7 +118,17 @@ class FrameWork
bool m_isRedrawEnabled;
double m_metresMinWidth;
int m_minRulerWidth;
bool m_doCenterViewport;
enum TGpsCenteringMode
{
EDoNothing,
ECenterAndScale,
ECenterOnly
};
TGpsCenteringMode m_centeringMode;
LocationRetrievedCallbackT m_locationObserver;
location::State m_locationState;
void AddRedrawCommandSure()
{
@ -155,6 +165,35 @@ class FrameWork
m_model.RemoveMap(datFile);
}
void 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();
m_locationState.UpdateGps(info);
if (m_centeringMode == ECenterAndScale)
{
CenterAndScaleViewport();
m_centeringMode = ECenterOnly;
}
else if (m_centeringMode == ECenterOnly)
CenterViewport(m_locationState.Position());
UpdateNow();
}
}
void OnCompassUpdate(location::CompassInfo const & info)
{
if (info.m_timestamp < location::POSITION_TIMEOUT_SECONDS)
{
m_locationState.UpdateCompass(info);
UpdateNow();
}
}
public:
FrameWork(shared_ptr<WindowHandle> windowHandle,
size_t bottomShift)
@ -163,10 +202,6 @@ public:
m_isBenchmarkInitialized(false),
m_currentBenchmark(0),
m_bgColor(0xEE, 0xEE, 0xDD, 0xFF),
// m_bgColor(0xF4 / 17 * 17, 0xFC / 17 * 17, 0xE8 / 17 * 17, 0xFF),
// m_bgColor(0xF2 / 17 * 17, 0xF2 / 17 * 17, 0xEE / 17 * 17, 0xFF),
// m_bgColor(0xF1 / 17 * 17, 0xF2 / 17 * 17, 0xEB / 17 * 17, 0xFF),
// m_bgColor(187, 187, 187, 255),
m_renderQueue(GetPlatform().SkinName(),
GetPlatform().IsMultiSampled(),
GetPlatform().DoPeriodicalUpdate(),
@ -177,7 +212,7 @@ public:
m_isRedrawEnabled(true),
m_metresMinWidth(20),
m_minRulerWidth(97),
m_doCenterViewport(false)
m_centeringMode(EDoNothing)
{
m_informationDisplay.setBottomShift(bottomShift);
#ifdef DRAW_TOUCH_POINTS
@ -204,6 +239,12 @@ public:
m_informationDisplay.setVisualScale(GetPlatform().VisualScale());
m_renderQueue.AddWindowHandle(m_windowHandle);
// initialize gps and compass subsystem
GetLocationService().SetGpsObserver(
boost::bind(&this_type::OnGpsUpdate, this, _1));
GetLocationService().SetCompassObserver(
boost::bind(&this_type::OnCompassUpdate, this, _1));
}
void InitBenchmark()
@ -246,30 +287,24 @@ public:
LOG(LINFO, ("Storage initialized"));
}
void InitLocator(shared_ptr<Locator> const & locator)
void StartLocationService(LocationRetrievedCallbackT observer)
{
m_locator = locator;
m_locator->addOnUpdateLocationFn(bind(&this_type::UpdateLocation, this, _1, _2, _3, _4));
m_locator->addOnUpdateHeadingFn(bind(&this_type::UpdateHeading, this, _1, _2, _3));
m_locator->addOnChangeModeFn(bind(&this_type::ChangeLocatorMode, this, _1, _2));
// m_locator->start(Locator::ERoughMode);
m_locationObserver = observer;
m_centeringMode = ECenterAndScale;
// by default, we always start in accurate mode
GetLocationService().StartUpdate(true);
}
void StartLocator(Locator::EMode mode)
void StopLocationService()
{
if (mode == Locator::EPreciseMode)
m_doCenterViewport = true;
}
void StopLocator()
{
m_informationDisplay.enablePosition(false);
m_informationDisplay.enableHeading(false);
m_doCenterViewport = false;
// reset callback
m_locationObserver.clear();
m_centeringMode = EDoNothing;
GetLocationService().StopUpdate();
m_locationState.TurnOff();
Invalidate();
}
bool IsEmptyModel()
{
return m_model.GetWorldRect() == m2::RectD::GetEmptyRect();
@ -352,16 +387,10 @@ public:
void SetOrientation(EOrientation orientation)
{
m_navigator.SetOrientation(orientation);
m_informationDisplay.setOrientation(orientation);
m_locationState.SetOrientation(orientation);
UpdateNow();
}
/// By VNG: I think, you don't need such selectors!
//ScreenBase const & Screen() const
//{
// return m_navigator.Screen();
//}
int GetCurrentScale() const
{
m2::PointD textureCenter(m_renderQueue.renderState().m_textureWidth / 2,
@ -487,6 +516,8 @@ public:
m_informationDisplay.doDraw(pDrawer);
InformationDisplay::DrawMyPosition(*pDrawer, m_navigator.Screen(), m_locationState);
e->drawer()->screen()->endFrame();
OGLCHECK(glPopMatrix());
@ -494,44 +525,6 @@ public:
}
}
/* void DisableMyPositionAndHeading()
{
m_informationDisplay.enablePosition(false);
m_informationDisplay.enableHeading(false);
UpdateNow();
}*/
void UpdateLocation(m2::PointD const & mercatorPos, double errorRadius, double locTimeStamp, double curTimeStamp)
{
m_informationDisplay.setPosition(mercatorPos, errorRadius);
if ((m_locator->mode() == Locator::EPreciseMode)
&& (curTimeStamp - locTimeStamp >= 0)
&& (curTimeStamp - locTimeStamp < 60 * 5))
{
if (m_doCenterViewport)
{
m_informationDisplay.setLocatorMode(Locator::EPreciseMode);
CenterAndScaleViewport();
m_doCenterViewport = false;
}
else
CenterViewport(mercatorPos);
}
UpdateNow();
}
void ChangeLocatorMode(Locator::EMode oldMode, Locator::EMode newMode)
{
if (newMode != Locator::EPreciseMode)
m_informationDisplay.setLocatorMode(newMode);
/// in precise mode we'll update informationDisplay.locatorMode on the
/// first "fresh" position recieved.
m_doCenterViewport = true;
Invalidate();
}
void CenterViewport(m2::PointD const & pt)
{
m_navigator.CenterViewport(pt);
@ -561,46 +554,41 @@ public:
{
}
/// @TODO refactor to accept point and min visible length
void CenterAndScaleViewport()
{
m2::PointD pt = m_informationDisplay.position();
m2::PointD const pt = m_locationState.Position();
m_navigator.CenterViewport(pt);
m2::RectD ClipRect = m_navigator.Screen().ClipRect();
m2::RectD clipRect = m_navigator.Screen().ClipRect();
double errorRadius = m_informationDisplay.errorRadius();
double xMinSize = 6 * max(errorRadius, MercatorBounds::ConvertMetresToX(pt.x, m_metresMinWidth));
double yMinSize = 6 * max(errorRadius, MercatorBounds::ConvertMetresToY(pt.y, m_metresMinWidth));
double const xMinSize = 6 * max(m_locationState.ErrorRadius(),
MercatorBounds::ConvertMetresToX(pt.x, m_metresMinWidth));
double const yMinSize = 6 * max(m_locationState.ErrorRadius(),
MercatorBounds::ConvertMetresToY(pt.y, m_metresMinWidth));
bool needToScale = false;
if (ClipRect.SizeX() < ClipRect.SizeY())
needToScale = ClipRect.SizeX() > xMinSize * 3;
if (clipRect.SizeX() < clipRect.SizeY())
needToScale = clipRect.SizeX() > xMinSize * 3;
else
needToScale = ClipRect.SizeY() > yMinSize * 3;
needToScale = clipRect.SizeY() > yMinSize * 3;
/* if ((ClipRect.SizeX() < 3 * errorRadius) || (ClipRect.SizeY() < 3 * errorRadius))
needToScale = true;*/
if (needToScale)
{
double k = max(xMinSize / ClipRect.SizeX(),
yMinSize / ClipRect.SizeY());
double const k = max(xMinSize / clipRect.SizeX(),
yMinSize / clipRect.SizeY());
ClipRect.Scale(k);
m_navigator.SetFromRect(ClipRect);
clipRect.Scale(k);
m_navigator.SetFromRect(clipRect);
}
UpdateNow();
}
void UpdateHeading(double trueHeading, double magneticHeading, double accuracy)
{
m_informationDisplay.setHeading(trueHeading, magneticHeading, accuracy);
Invalidate();
}
/// Show all model by it's world rect.
void ShowAll()
{
@ -642,14 +630,12 @@ public:
}
void DoDrag(DragEvent const & e)
{
if (m_locator && (m_locator->mode() == Locator::EPreciseMode) && (!m_doCenterViewport))
m_locator->setMode(Locator::ERoughMode);
m_centeringMode = EDoNothing;
m2::PointD ptShift = m_renderQueue.renderState().coordSystemShift(true);
m2::PointD pos = m_navigator.OrientPoint(e.Pos()) + ptShift;
m_navigator.DoDrag(pos,
GetPlatform().TimeInSec());
m_navigator.DoDrag(pos, GetPlatform().TimeInSec());
#ifdef DRAW_TOUCH_POINTS
m_informationDisplay.setDebugPoint(0, pos);
@ -657,6 +643,7 @@ public:
Invalidate();
}
void StopDrag(DragEvent const & e)
{
m2::PointD ptShift = m_renderQueue.renderState().coordSystemShift(true);
@ -685,16 +672,11 @@ public:
//@{
void ScaleToPoint(ScaleToPointEvent const & e)
{
m2::PointD ptShift = m_renderQueue.renderState().coordSystemShift(true);
m2::PointD const pt = (m_centeringMode == EDoNothing)
? m_navigator.OrientPoint(e.Pt()) + m_renderQueue.renderState().coordSystemShift(true)
: m_navigator.Screen().PixelRect().Center();
m2::PointD pt = m_navigator.OrientPoint(e.Pt()) + ptShift;
if ((m_locator) && (m_locator->mode() == Locator::EPreciseMode))
pt = m_navigator.Screen().PixelRect().Center();
m_navigator.ScaleToPoint(pt,
e.ScaleFactor(),
GetPlatform().TimeInSec());
m_navigator.ScaleToPoint(pt, e.ScaleFactor(), GetPlatform().TimeInSec());
UpdateNow();
}
@ -717,7 +699,7 @@ public:
m2::PointD pt1 = m_navigator.OrientPoint(e.Pt1()) + ptShift;
m2::PointD pt2 = m_navigator.OrientPoint(e.Pt2()) + ptShift;
if ((m_locator) && (m_locator->mode() == Locator::EPreciseMode) && (!m_doCenterViewport))
if ((m_locationState & location::State::EGps) && (m_centeringMode == ECenterOnly))
{
m2::PointD ptC = (pt1 + pt2) / 2;
m2::PointD ptDiff = m_navigator.Screen().PixelRect().Center() - ptC;
@ -742,7 +724,7 @@ public:
m2::PointD pt1 = m_navigator.OrientPoint(e.Pt1()) + ptShift;
m2::PointD pt2 = m_navigator.OrientPoint(e.Pt2()) + ptShift;
if ((m_locator) && (m_locator->mode() == Locator::EPreciseMode) && (!m_doCenterViewport))
if ((m_locationState & location::State::EGps) && (m_centeringMode == ECenterOnly))
{
m2::PointD ptC = (pt1 + pt2) / 2;
m2::PointD ptDiff = m_navigator.Screen().PixelRect().Center() - ptC;
@ -767,8 +749,7 @@ public:
m2::PointD pt1 = m_navigator.OrientPoint(e.Pt1()) + ptShift;
m2::PointD pt2 = m_navigator.OrientPoint(e.Pt2()) + ptShift;
if ((m_locator) && (m_locator->mode() == Locator::EPreciseMode) && (!m_doCenterViewport))
if ((m_locationState & location::State::EGps) && (m_centeringMode == ECenterOnly))
{
m2::PointD ptC = (pt1 + pt2) / 2;
m2::PointD ptDiff = m_navigator.Screen().PixelRect().Center() - ptC;
@ -787,5 +768,3 @@ public:
}
//@}
};
#include "../base/stop_mem_debug.hpp"

View file

@ -2,6 +2,7 @@
#include "information_display.hpp"
#include "drawer_yg.hpp"
#include "location_state.hpp"
#include "../indexer/mercator.hpp"
#include "../yg/defines.hpp"
@ -15,12 +16,12 @@
#include "../base/logging.hpp"
#include "../base/mutex.hpp"
#include <boost/bind.hpp>
using namespace location;
InformationDisplay::InformationDisplay()
: m_headingOrientation(-math::pi / 2),
m_mode(Locator::ERoughMode)
{
enablePosition(false);
enableHeading(false);
enableDebugPoints(false);
enableRuler(false);
enableCenter(false);
@ -44,122 +45,49 @@ void InformationDisplay::setBottomShift(double bottomShift)
m_bottomShift = bottomShift;
}
void InformationDisplay::setOrientation(EOrientation orientation)
{
switch (orientation)
{
case EOrientation0:
m_headingOrientation = -math::pi / 2;
break;
case EOrientation90:
m_headingOrientation = math::pi;
break;
case EOrientation180:
m_headingOrientation = math::pi / 2;
break;
case EOrientation270:
m_headingOrientation = 0;
break;
}
}
void InformationDisplay::setDisplayRect(m2::RectI const & rect)
{
m_displayRect = rect;
}
void InformationDisplay::enablePosition(bool doEnable)
void InformationDisplay::DrawMyPosition(DrawerYG & drawer,
ScreenBase const & screen,
location::State const & state)
{
m_isPositionEnabled = doEnable;
}
double pxErrorRadius;
m2::PointD pxPosition;
if ((state & State::EGps) || (state & State::ECompass))
{
pxPosition = screen.GtoP(state.Position());
pxErrorRadius = pxPosition.Length(screen.GtoP(state.Position()
+ m2::PointD(state.ErrorRadius(), 0)));
}
void InformationDisplay::setPosition(m2::PointD const & mercatorPos, double errorRadius)
{
enablePosition(true);
m_position = mercatorPos;
m_errorRadius = errorRadius;
}
m2::PointD const & InformationDisplay::position() const
{
return m_position;
}
double InformationDisplay::errorRadius() const
{
return m_errorRadius;
}
void InformationDisplay::drawPosition(DrawerYG * pDrawer)
{
/// Drawing position and heading
m2::PointD pxPosition = m_screen.GtoP(m_position);
pDrawer->drawSymbol(pxPosition, "current-position", yg::EPosCenter, yg::maxDepth);
double pxErrorRadius = pxPosition.Length(m_screen.GtoP(m_position + m2::PointD(m_errorRadius, 0)));
// pDrawer->screen()->drawArc(pxPosition, 0, math::pi * 2, pxErrorRadius, yg::Color(0, 0, 255, m_mode == Locator::EPreciseMode ? 64 : 32), yg::maxDepth - 2);
pDrawer->screen()->fillSector(pxPosition, 0, math::pi * 2, pxErrorRadius, yg::Color(0, 0, 255, m_mode == Locator::EPreciseMode ? 64 : 32), yg::maxDepth - 3);
}
void InformationDisplay::setLocatorMode(Locator::EMode mode)
{
m_mode = mode;
}
void InformationDisplay::enableHeading(bool doEnable)
{
m_isHeadingEnabled = doEnable;
}
void InformationDisplay::setHeading(double trueHeading, double magneticHeading, double accuracy)
{
enableHeading(true);
m_trueHeading = trueHeading;
m_magneticHeading = magneticHeading;
m_headingAccuracy = accuracy;
}
void InformationDisplay::drawHeading(DrawerYG *pDrawer)
{
if (m_mode == Locator::ERoughMode)
return;
double trueHeadingRad = m_trueHeading / 180 * math::pi;
double headingAccuracyRad = m_headingAccuracy / 180 * math::pi;
m2::PointD pxPosition = m_screen.GtoP(m_position);
double pxErrorRadius = pxPosition.Length(m_screen.GtoP(m_position + m2::PointD(m_errorRadius, 0)));
/// true heading
pDrawer->screen()->drawSector(pxPosition,
trueHeadingRad + m_headingOrientation - headingAccuracyRad,
trueHeadingRad + m_headingOrientation + headingAccuracyRad,
pxErrorRadius,
yg::Color(255, 255, 255, 192),
yg::maxDepth);
pDrawer->screen()->fillSector(pxPosition,
trueHeadingRad + m_headingOrientation - headingAccuracyRad,
trueHeadingRad + m_headingOrientation + headingAccuracyRad,
pxErrorRadius,
yg::Color(255, 255, 255, 96),
yg::maxDepth - 1);
/* /// magnetic heading
double magneticHeadingRad = m_magneticHeading / 180 * math::pi;
pDrawer->screen()->drawSector(pxPosition,
magneticHeadingRad + m_headingOrientation - headingAccuracyRad,
magneticHeadingRad + m_headingOrientation + headingAccuracyRad,
pxErrorRadius,
yg::Color(0, 255, 0, 64),
yg::maxDepth);
pDrawer->screen()->fillSector(pxPosition,
magneticHeadingRad + m_headingOrientation - headingAccuracyRad,
magneticHeadingRad + m_headingOrientation + headingAccuracyRad,
pxErrorRadius,
yg::Color(0, 255, 0, 32),
yg::maxDepth - 1);
*/
if (state & State::EGps)
{
// my position symbol
drawer.drawSymbol(pxPosition, "current-position", yg::EPosCenter, yg::maxDepth);
// my position circle
drawer.screen()->fillSector(pxPosition, 0, math::pi * 2, pxErrorRadius,
yg::Color(0, 0, 255, (state & State::EPreciseMode) ? 32 : 16),
yg::maxDepth - 3);
// display compass only if position is available
if (state & State::ECompass)
{
drawer.screen()->drawSector(pxPosition,
state.Heading() - state.HeadingAccuracy(),
state.Heading() + state.HeadingAccuracy(),
pxErrorRadius,
yg::Color(255, 255, 255, 192),
yg::maxDepth);
drawer.screen()->fillSector(pxPosition,
state.Heading() - state.HeadingAccuracy(),
state.Heading() + state.HeadingAccuracy(),
pxErrorRadius,
yg::Color(255, 255, 255, 96),
yg::maxDepth - 1);
}
}
}
void InformationDisplay::enableDebugPoints(bool doEnable)
@ -644,10 +572,6 @@ void InformationDisplay::drawBenchmarkInfo(DrawerYG * pDrawer)
void InformationDisplay::doDraw(DrawerYG *drawer)
{
m_yOffset = 0;
if (m_isPositionEnabled)
drawPosition(drawer);
if (m_isHeadingEnabled)
drawHeading(drawer);
if (m_isDebugPointsEnabled)
drawDebugPoints(drawer);
if (m_isRulerEnabled)

View file

@ -1,13 +1,17 @@
#pragma once
#include "window_handle.hpp"
#include "locator.hpp"
#include "../geometry/point2d.hpp"
#include "../geometry/rect2d.hpp"
#include "../geometry/screenbase.hpp"
#include "../base/timer.hpp"
#include "../base/logging.hpp"
namespace location
{
class State;
}
class DrawerYG;
/// Class, which displays additional information on the primary layer.
@ -20,18 +24,6 @@ private:
m2::RectI m_displayRect;
int m_yOffset;
double m_headingOrientation;
bool m_isHeadingEnabled;
double m_trueHeading;
double m_magneticHeading;
double m_headingAccuracy;
bool m_isPositionEnabled;
Locator::EMode m_mode;
m2::PointD m_position;
double m_errorRadius;
/// for debugging purposes
/// up to 10 debugging points
bool m_isDebugPointsEnabled;
@ -81,22 +73,14 @@ public:
InformationDisplay();
static void DrawMyPosition(DrawerYG & drawer,
ScreenBase const & screen,
location::State const & state);
void setScreen(ScreenBase const & screen);
void setDisplayRect(m2::RectI const & rect);
void setBottomShift(double bottomShift);
void setVisualScale(double visualScale);
void setOrientation(EOrientation orientation);
void enablePosition(bool doEnable);
void setPosition(m2::PointD const & mercatorPos, double errorRadius);
void setLocatorMode(Locator::EMode mode);
m2::PointD const & position() const;
double errorRadius() const;
void drawPosition(DrawerYG * pDrawer);
void enableHeading(bool doEnable);
void setHeading(double trueHeading, double magneticHeading, double accuracy);
void drawHeading(DrawerYG * pDrawer);
void enableDebugPoints(bool doEnable);
void setDebugPoint(int pos, m2::PointD const & pt);

66
map/location_state.cpp Normal file
View file

@ -0,0 +1,66 @@
#include "location_state.hpp"
#include "../platform/location.hpp"
#include "../indexer/mercator.hpp"
namespace location
{
State::State() : m_deviceOrientation(-math::pi / 2), m_type(ENone)
{
}
void State::UpdateGps(GpsInfo const & info)
{
if (info.m_status == EAccurateMode
|| info.m_status == ERoughMode)
{
m_type |= EGps;
if (info.m_status == EAccurateMode)
m_type |= EPreciseMode;
else
m_type &= !EPreciseMode;
m_positionMercator = m2::PointD(MercatorBounds::LonToX(info.m_longitude),
MercatorBounds::LatToY(info.m_latitude));
m2::RectD const errorRectXY =
MercatorBounds::MetresToXY(info.m_longitude, info.m_latitude,
info.m_horizontalAccuracy);
m_errorRadiusMercator = sqrt((errorRectXY.SizeX() * errorRectXY.SizeX()
+ errorRectXY.SizeY() * errorRectXY.SizeY()) / 4);
}
else
{
m_type &= !EGps;
}
}
void State::UpdateCompass(CompassInfo const & info)
{
m_type |= ECompass;
m_headingRad = ((info.m_trueHeading >= 0.0) ? info.m_trueHeading : info.m_magneticHeading)
/ 180 * math::pi;
m_headingAccuracyRad = info.m_accuracy / 180 * math::pi;
}
void State::SetOrientation(EOrientation orientation)
{
switch (orientation)
{
case EOrientation0:
m_deviceOrientation = -math::pi / 2;
break;
case EOrientation90:
m_deviceOrientation = math::pi;
break;
case EOrientation180:
m_deviceOrientation = math::pi / 2;
break;
case EOrientation270:
m_deviceOrientation = 0;
break;
}
}
}

55
map/location_state.hpp Normal file
View file

@ -0,0 +1,55 @@
#pragma once
#include "../geometry/point2d.hpp"
#include "../geometry/screenbase.hpp"
namespace location
{
class GpsInfo;
class CompassInfo;
class State
{
double m_errorRadiusMercator;
m2::PointD m_positionMercator;
double m_deviceOrientation;
double m_headingRad;
double m_headingAccuracyRad;
public:
enum SymbolType
{
ENone = 0x0,
EGps = 0x1,
EPreciseMode = 0x2,
ECompass = 0x4,
};
State();
/// @return GPS error radius in mercator
double ErrorRadius() const { return m_errorRadiusMercator; }
/// @return GPS center point in mercator
m2::PointD Position() const { return m_positionMercator; }
/// takes into account device's orientation
/// @return angle in radians
double Heading() const { return m_deviceOrientation + m_headingRad; }
/// @return angle in radians
double HeadingAccuracy() const { return m_headingAccuracyRad; }
void TurnOff() { m_type = ENone; }
void UpdateGps(GpsInfo const & info);
void UpdateCompass(CompassInfo const & info);
void SetOrientation(EOrientation orientation);
operator int() const
{
return m_type;
}
private:
/// stores flags from SymbolType
int m_type;
};
}

View file

@ -1,54 +0,0 @@
#include "../base/SRC_FIRST.hpp"
#include "locator.hpp"
Locator::Locator() : m_isRunning(false)
{}
Locator::~Locator()
{}
void Locator::setMode(EMode mode)
{
m_mode = mode;
}
Locator::EMode Locator::mode() const
{
return m_mode;
}
void Locator::start(Locator::EMode /*mode*/)
{
m_isRunning = true;
}
void Locator::stop()
{
m_isRunning = false;
}
bool Locator::isRunning() const
{
return m_isRunning;
}
void Locator::callOnChangeModeFns(EMode oldMode, EMode newMode)
{
list<onChangeModeFn> handlers = m_onChangeModeFns;
for (list<onChangeModeFn>::iterator it = handlers.begin(); it != handlers.end(); ++it)
(*it)(oldMode, newMode);
}
void Locator::callOnUpdateLocationFns(m2::PointD const & pt, double errorRadius, double locTimeStamp, double curTimeStamp)
{
list<onUpdateLocationFn> handlers = m_onUpdateLocationFns;
for (list<onUpdateLocationFn>::iterator it = handlers.begin(); it != handlers.end(); ++it)
(*it)(pt, errorRadius, locTimeStamp, curTimeStamp);
}
void Locator::callOnUpdateHeadingFns(double trueHeading, double magneticHeading, double accuracy)
{
list<onUpdateHeadingFn> handlers = m_onUpdateHeadingFns;
for (list<onUpdateHeadingFn>::iterator it = handlers.begin(); it != handlers.end(); ++it)
(*it)(trueHeading, magneticHeading, accuracy);
}

View file

@ -1,64 +0,0 @@
#pragma once
#include "../std/list.hpp"
#include "../std/function.hpp"
#include "../geometry/point2d.hpp"
/// Base class for OS-dependent implementation of locator
class Locator
{
public:
enum EMode
{
ERoughMode, //< monitor only significant place changes, saves battery.
EPreciseMode //< monitor place changes with the maximum available accuracy
};
typedef function<void (m2::PointD const & /*latLongPos*/, double /*errorRadius*/, double /*locationTimeStamp*/, double /*currentTimeStamp*/)> onUpdateLocationFn;
typedef function<void (double /*trueHeading*/, double /*magneticHeading*/, double /*accuracy*/)> onUpdateHeadingFn;
typedef function<void (EMode /*oldMode*/, EMode /*newMode*/)> onChangeModeFn;
private:
list<onUpdateLocationFn> m_onUpdateLocationFns;
list<onUpdateHeadingFn> m_onUpdateHeadingFns;
list<onChangeModeFn> m_onChangeModeFns;
EMode m_mode;
bool m_isRunning;
public:
Locator();
virtual ~Locator();
bool isRunning() const;
virtual void start(EMode mode) = 0;
virtual void stop() = 0;
virtual void setMode(EMode mode) = 0;
EMode mode() const;
template <typename Fn>
void addOnUpdateLocationFn(Fn fn)
{
m_onUpdateLocationFns.push_back(fn);
}
template <typename Fn>
void addOnUpdateHeadingFn(Fn fn)
{
m_onUpdateHeadingFns.push_back(fn);
}
template <typename Fn>
void addOnChangeModeFn(Fn fn)
{
m_onChangeModeFns.push_back(fn);
}
void callOnUpdateLocationFns(m2::PointD const & pt, double errorRadius, double locTimeStamp, double curTimeStamp);
void callOnUpdateHeadingFns(double trueHeading, double magneticHeading, double accuracy);
void callOnChangeModeFns(EMode oldMode, EMode newMode);
};

View file

@ -22,7 +22,7 @@ HEADERS += \
render_queue_routine.hpp \
information_display.hpp \
settings.hpp \
locator.hpp
location_state.hpp \
SOURCES += \
feature_vec_model.cpp \
@ -34,7 +34,7 @@ SOURCES += \
render_queue_routine.cpp \
information_display.cpp \
settings.cpp \
locator.cpp
location_state.cpp \
!iphone*:!bada* {
HEADERS += qgl_render_context.hpp

View file

@ -35,6 +35,7 @@ SUBDIRS = 3party \
base \
coding \
geometry \
platform \
yg \
indexer \
version \

View file

@ -41,7 +41,12 @@ public:
void OnLocationUpdate(GpsInfo & newLocation)
{
newLocation.m_status = m_status;
NotifySubscribers(newLocation);
NotifyGpsObserver(newLocation);
}
void OnHeadingUpdate(CompassInfo & newHeading)
{
NotifyCompassObserver(newHeading);
}
// virtual bool IsServiceSupported()
@ -71,7 +76,7 @@ public:
m_status = EDisabledByUser;
GpsInfo info;
info.m_status = m_status;
NotifySubscribers(info);
NotifyGpsObserver(info);
}
else
{
@ -79,23 +84,21 @@ public:
{
m_status = EAccurateMode;
m_locationManager.desiredAccuracy = kCLLocationAccuracyBest;
// also enable compass
#ifdef OMIM_OS_IPHONE
if ([CLLocationManager headingAvailable])
[m_locationManager startHeadingUpdate];
#endif
}
else
{
m_status = ERoughMode;
m_locationManager.desiredAccuracy = ROUGH_ACCURACY;
// also disable compass
#ifdef OMIM_OS_IPHONE
if ([CLLocationManager headingAvailable])
[m_locationManager stopHeadingUpdate];
#endif
}
[m_locationManager startUpdatingLocation];
// enable compass
#ifdef OMIM_OS_IPHONE
if ([CLLocationManager headingAvailable])
{
m_locationManager.headingFilter = 1.0;
[m_locationManager startUpdatingHeading];
}
#endif
}
}
@ -103,7 +106,7 @@ public:
{
#ifdef OMIM_OS_IPHONE
if ([CLLocationManager headingAvailable])
[m_locationManager stopHeadingUpdate];
[m_locationManager stopUpdatingHeading];
#endif
[m_locationManager stopUpdatingLocation];
}
@ -134,7 +137,7 @@ public:
info.m_verticalAccuracy = location.verticalAccuracy;
info.m_latitude = location.coordinate.latitude;
info.m_longitude = location.coordinate.longitude;
info.m_timestamp = [location.timestamp timeIntervalSince1970];
info.m_timestamp = -[location.timestamp timeIntervalSinceNow];
}
- (void)locationManager:(CLLocationManager *)manager
@ -146,6 +149,22 @@ public:
m_service->OnLocationUpdate(newInfo);
}
#ifdef OMIM_OS_IPHONE
- (void)locationManager:(CLLocationManager *)manager
didUpdateHeading:(CLHeading *)newHeading
{
CompassInfo newInfo;
newInfo.m_magneticHeading = newHeading.magneticHeading;
newInfo.m_trueHeading = newHeading.trueHeading;
newInfo.m_accuracy = newHeading.headingAccuracy;
newInfo.m_x = newHeading.x;
newInfo.m_y = newHeading.y;
newInfo.m_z = newHeading.z;
newInfo.m_timestamp = -[newHeading.timestamp timeIntervalSinceNow];
m_service->OnHeadingUpdate(newInfo);
}
#endif
- (void)locationManager:(CLLocationManager *)manager
didFailWithError:(NSError *)error
{

View file

@ -1,52 +0,0 @@
#include "location.hpp"
#include "../std/algorithm.hpp"
#include <boost/bind.hpp>
namespace location
{
void LocationService::NotifySubscribers(GpsInfo const & info)
{
for (GpsObserversT::iterator it = m_gpsObservers.begin();
it != m_gpsObservers.end(); ++it)
(*it)(info);
}
void LocationService::NotifySubscribers(CompassInfo const & info)
{
for (CompassObserversT::iterator it = m_compassObservers.begin();
it != m_compassObservers.end(); ++it)
(*it)(info);
}
void LocationService::SubscribeToGpsUpdates(TGpsCallback observer)
{
// if (std::find(m_gpsObservers.begin(), m_gpsObservers.end(), boost::bind(&observer))
// == m_gpsObservers.end())
m_gpsObservers.push_back(observer);
}
void LocationService::SubscribeToCompassUpdates(TCompassCallback observer)
{
// if (std::find(m_compassObservers.begin(), m_compassObservers.end(), observer)
// == m_compassObservers.end())
m_compassObservers.push_back(observer);
}
// void LocationService::Unsubscribe(TGpsCallback observer)
// {
// GpsObserversT::iterator found =
// std::find(m_gpsObservers.begin(), m_gpsObservers.end(), observer);
// if (found != m_gpsObservers.end())
// m_gpsObservers.erase(found);
// }
// void LocationService::Unsubscribe(TCompassCallback observer)
// {
// CompassObserversT::iterator found =
// std::find(m_compassObservers.begin(), m_compassObservers.end(), observer);
// if (found != m_compassObservers.end())
// m_compassObservers.erase(found);
// }
}

View file

@ -7,6 +7,9 @@
namespace location
{
/// after this period we cont position as "too old"
static double const POSITION_TIMEOUT_SECONDS = 300.0;
enum TLocationStatus
{
ENotSupported, //!< GpsInfo fields are not valid with this value
@ -19,9 +22,9 @@ namespace location
struct GpsInfo
{
TLocationStatus m_status;
double m_timestamp; //!< seconds from 01/01/1970
double m_latitude; //!< degrees @TODO mercator
double m_longitude; //!< degrees @TODO mercator
double m_timestamp; //!< how many seconds ago the position was retrieved
double m_latitude; //!< degrees
double m_longitude; //!< degrees
double m_horizontalAccuracy; //!< metres
double m_altitude; //!< metres
double m_verticalAccuracy; //!< metres
@ -29,9 +32,10 @@ namespace location
double m_speed; //!< metres per second
};
/// @note always check m_status before using this structure
struct CompassInfo
{
double m_timestamp; //!< seconds from 01/01/1970
double m_timestamp; //!< how many seconds ago the heading was retrieved
double m_magneticHeading; //!< positive degrees from the magnetic North
double m_trueHeading; //!< positive degrees from the true North
double m_accuracy; //!< offset from magnetic to true North
@ -40,29 +44,39 @@ namespace location
int m_z;
};
typedef boost::function1<void, GpsInfo const &> TGpsCallback;
typedef boost::function1<void, CompassInfo const &> TCompassCallback;
typedef boost::function<void (GpsInfo const &)> TGpsCallback;
typedef boost::function<void (CompassInfo const &)> TCompassCallback;
class LocationService
{
typedef vector<TGpsCallback> GpsObserversT;
GpsObserversT m_gpsObservers;
typedef vector<TCompassCallback> CompassObserversT;
CompassObserversT m_compassObservers;
TGpsCallback m_gpsObserver;
TCompassCallback m_compassObserver;
protected:
void NotifySubscribers(GpsInfo const & info);
void NotifySubscribers(CompassInfo const & info);
void NotifyGpsObserver(GpsInfo const & info)
{
if (m_gpsObserver)
m_gpsObserver(info);
}
void NotifyCompassObserver(CompassInfo const & info)
{
if (m_compassObserver)
m_compassObserver(info);
}
public:
/// @note unsubscribe doesn't work with boost::function
void SubscribeToGpsUpdates(TGpsCallback observer);
void SubscribeToCompassUpdates(TCompassCallback observer);
// void Unsubscribe(TGpsCallback observer);
// void Unsubscribe(TCompassCallback observer);
void SetGpsObserver(TGpsCallback observer)
{
m_gpsObserver = observer;
}
void SetCompassObserver(TCompassCallback observer)
{
m_compassObserver = observer;
}
/// to change active accuracy mode just call it again
/// @param useAccurateMode if true also enables compass if it's available
/// @note also enables compass if it's available
virtual void StartUpdate(bool useAccurateMode) = 0;
virtual void StopUpdate() = 0;
};

View file

@ -11,17 +11,20 @@ include($$ROOT_DIR/common.pri)
QT *= core network
SOURCES += \
qtplatform.cpp \
qt_download_manager.cpp \
qt_download.cpp \
location.cpp \
!iphone* {
SOURCES += \
qtplatform.cpp \
qt_download_manager.cpp \
qt_download.cpp \
HEADERS += \
qt_download_manager.hpp \
qt_download.hpp \
}
HEADERS += \
platform.hpp \
download_manager.hpp \
qt_download_manager.hpp \
qt_download.hpp \
location.hpp \
mac|iphone* {

View file

@ -14,6 +14,11 @@ win32 {
LIBS += -lShell32
}
mac {
LIBS += -framework CoreLocation -framework Foundation
}
SOURCES += \
../../testing/testingmain.cpp \
platform_test.cpp \

View file

@ -71,6 +71,16 @@ namespace qt
// m_framework.ShowFeature(p);
//}
void DrawWidget::OnEnableMyPosition(LocationRetrievedCallbackT observer)
{
m_framework.StartLocationService(observer);
}
void DrawWidget::OnDisableMyPosition()
{
m_framework.StopLocationService();
}
void DrawWidget::MoveLeft()
{
m_framework.Move(math::pi, 0.5);

View file

@ -56,6 +56,9 @@ namespace qt
void SetScaleControl(QSlider * pScale);
void OnEnableMyPosition(LocationRetrievedCallbackT observer);
void OnDisableMyPosition();
//model_t * GetModel() { return &(m_framework.get_model()); }
//void ShowFeature(Feature const & p);

View file

@ -232,6 +232,13 @@ void MainWindow::CreateNavigationBar()
pBar->setIconSize(QSize(32, 32));
{
// add my position button with "checked" behavior
m_pMyPosition = pBar->addAction(QIcon(":/navig64/plus.png"),
tr("My Position"),
this,
SLOT(OnMyPosition()));
m_pMyPosition->setCheckable(true);
// add view actions 1
button_t arr[] = {
{ tr("Left"), ":/navig64/left.png", SLOT(MoveLeft()) },
@ -333,4 +340,23 @@ void MainWindow::OnPreferences()
Settings::Set("AutomaticUpdateCheck", autoUpdatesEnabled);
}
void MainWindow::OnLocationFound()
{
// @TODO change button icon to "found location"
}
void MainWindow::OnMyPosition()
{
if (m_pMyPosition->isChecked())
{
// @TODO change button icon to "searching location"
m_pDrawWidget->OnEnableMyPosition(boost::bind(&MainWindow::OnLocationFound, this));
}
else
{
// @TODO change
m_pDrawWidget->OnDisableMyPosition();
}
}
}

View file

@ -16,6 +16,7 @@ namespace qt
class MainWindow : public QMainWindow
{
QAction * m_pMyPosition;
DrawWidget * m_pDrawWidget;
QDockWidget * m_Docks[2];
//FindTableWnd * m_pFindTable;
@ -34,6 +35,9 @@ namespace qt
void SaveState();
void LoadState();
private:
void OnLocationFound();
protected:
void CreatePanelImpl(size_t i, QString const & name, QKeySequence const & hotkey,
char const * slot);
@ -56,6 +60,7 @@ namespace qt
void ShowGuidePanel();
void OnAbout();
void OnPreferences();
void OnMyPosition();
};
}

View file

@ -15,6 +15,8 @@ win32 {
}
macx {
LIBS += -framework CoreLocation -framework Foundation
ICON = res/mac.icns
PLIST_FILE = Info.plist
# path to original plist, which will be processed by qmake and later by us