Android library project that intends to simplify the usage of location providers and activity recognition with a nice fluid API.
Supported Android versions: Android 4.0+
You should add this to your dependencies:
implement 'io.nlopez.smartlocation:library:3.3.3'
Google Play Services compatible version: 11.4.2
If you want the rxjava wrappers, these are now in a separate dependency. Just add this new dependency as well:
implement 'io.nlopez.smartlocation:rx:3.3.3'
If you got any problem compiling, please check the Common Issues section at the bottom of this document.
For starting the location service:
SmartLocation.with(context).location()
.start(new OnLocationUpdatedListener() { ... });
If you just want to get a single location (not periodic) you can just use the oneFix modifier. Example:
SmartLocation.with(context).location()
.oneFix()
.start(new OnLocationUpdatedListener() { ... });
For stopping the location just use the stop method.
SmartLocation.with(context).location().stop();
You can get some information about the current status of location providers to know if you will be able to use the location providers.
// Check if the location services are enabled
SmartLocation.with(context).location().state().locationServicesEnabled();
// Check if any provider (network or gps) is enabled
SmartLocation.with(context).location().state().isAnyProviderAvailable();
// Check if GPS is available
SmartLocation.with(context).location().state().isGpsAvailable();
// Check if Network is available
SmartLocation.with(context).location().state().isNetworkAvailable();
// Check if the passive provider is available
SmartLocation.with(context).location().state().isPassiveAvailable();
// Check if the location is mocked
SmartLocation.with(context).location().state().isMockSettingEnabled();
There are three presets for location parameters:
LocationParams.BEST_EFFORT
(default)LocationParams.NAVIGATION
LocationParams.LAZY
You can change it (if you want one other than the default one) by using the config(locationParams)
modifier.
If you want to add some custom parameters for the distances or times involved in the location strategy, you can create your own LocationParams class.
There are some providers shipped with the library.
LocationGooglePlayServicesWithFallbackProvider
(default). This one will use the Fused Location Provider if it's present, or the LocationManager as fallback if it's not.LocationGooglePlayServicesProvider
This will use the Fused Location Provider.LocationManagerProvider
This is the legacy implementation that uses LocationManager.LocationBasedOnActivityProvider
This allows you to use the activity recognition system to modify the location strategy depending on the activity changes (if the user is walking, running, on a car, a bike...).MultiFallbackLocationProvider
This lets you create your own "fallback provider" if the underlying location service is not available. See "Multiple Fallback Provider" below for details.You can implement your own if you want. That's ideal if you wanted to use a mock one for testing or something like that, or add support to another possible provider.
Example:
SmartLocation.with(context).location(new LocationBasedOnActivityProvider(callback))
.start(new OnLocationUpdatedListener() { ... });
The MultiFallbackProvider
lets you create your own provider that utilizes multiple underlying location services.The provider will use the location services in the order in which they are added to its Builder
, which has convenience methods for setting up the Google Play Services provider and the default LocationManager
provider.Providers must implement the ServiceLocationProvider
interface to enable the fallback behavior.Example:
LocationProvider myProvider = new MyLocationProvider();
LocationProvider fallbackProvider = new MultiFallbackProvider.Builder()
.withGooglePlayServicesProvider().withProvider(myProvider).build();
For starting the activity recognition service, you should run:
SmartLocation.with(context).activityRecognition()
.start(new OnActivityUpdatedListener() { ... });
For stopping the activity recognition you could use the stop method.
SmartLocation.with(context).activityRecognition().stop();
We can add geofences and receive the information when we enter, exit or dwell in a Geofence. The geofences are defined by a GeofenceModel, and you should use the requestId as a identifier.
We can add and remove geofences with a similar syntax as all the others.
GeofenceModel mestalla = new GeofenceModel.Builder("id_mestalla")
.setTransition(Geofence.GEOFENCE_TRANSITION_ENTER)
.setLatitude(39.47453120000001)
.setLongitude(-0.358065799999963)
.setRadius(500)
.build();
GeofenceModel cuenca = new GeofenceModel.Builder("id_cuenca")
.setTransition(Geofence.GEOFENCE_TRANSITION_EXIT)
.setLatitude(40.0703925)
.setLongitude(-2.1374161999999615)
.setRadius(2000)
.build();
SmartLocation.with(context).geofencing()
.add(mestalla)
.add(cuenca)
.remove("already_existing_geofence_id")
.start(new OnGeofencingTransitionListener() { ... });
If you want to capture the Geofence transitions without the app running, you can hook up a BroadcastReceiver to the intent action stored in the GeofencingGooglePlayServicesProvider.BROADCAST_INTENT_ACTION
constant. The intent will come with the geofence, the location and the type of transition within the bundle.
The library has support for direct geocoding (aka getting a Location object based on a String) and reverse geocoding (getting the Street name based on a Location object).
There are pretty basic calls in the API for both operations separatedly.
SmartLocation.with(context).geocoding()
.direct("Estadi de Mestalla", new OnGeocodingListener() {
@Override
public void onLocationResolved(String name, List<LocationAddress> results) {
// name is the same you introduced in the parameters of the call
// results could come empty if there is no match, so please add some checks around that
// LocationAddress is a wrapper class for Address that has a Location based on its data
if (results.size() > 0) {
Location mestallaLocation = results.get(0).getLocation();
// [...] Do your thing! :D
}
}
});
SmartLocation.with(context).geocoding()
.reverse(location, new OnReverseGeocodingListener() {
@Override
public onAddressResolved(Location original, List<Address> results) {
// ...
}
});
But we can mix and batch those requests, if needed. Also, you can provide the number of maximum possible matches you want to receive for each one of the lookups separatedly.
Location myLocation1 = new Location(...);
SmartLocation.with(context).geocoding()
.add("Estadi de Mestalla", 5)
.add("Big Ben", 2)
.add(myLocation1, 4)
.start(directGeocodingListener, reverseGeocodingListener);
This will launch a new call to the callbacks everytime one of the geofence lookups is resolved.
You should invoke the stop method whenever the calling activity/fragment or whatever is going to be destroyed, for cleanup purposes.
The wrappers to rxjava2 are located in this package.
implement 'io.nlopez.smartlocation:rx:3.3.1'
You can wrap the calls with ObservableFactory methods to retrieve an Observable object. You won't need to call start, just subscribe to the observable to get the updates.
For example, for location:
Observable<Location> locationObservable = ObservableFactory.from(SmartLocation.with(context).location());
locationObservable.subscribe(new Action1<Location>() {
@Override
public void call(Location location) {
// Do your stuff here :)
}
});
If you are already using Google Play Services in your project and have problems compiling, you can try setting the transitive property to false:
implement ('io.nlopez.smartlocation:library:3.3.3') {
transitive = false
}
If you got an error in the manifest merging, like this one:
> Manifest merger failed : Attribute meta-data#com.google.android.gms.version@value value=(@integer/google_play_services_version) from AndroidManifest.xml:44:13
is also present at io.nlopez.smartlocation:library:3.0.5:28:13 value=(6587000)
Suggestion: add 'tools:replace="android:value"' to <meta-data> element at AndroidManifest.xml:42:9 to override
Error:(46, 13) Attribute meta-data#com.google.android.gms.version@value value=(@integer/google_play_services_version) from AndroidManifest.xml:46:13
If you follow the suggestion provided, you can get rid of it easily. Just change in your manifest the meta-data tag with the google play services version, like this:
<meta-data tools:replace="android:value" android:name="com.google.android.gms.version" android:value="@integer/google_play_services_version" />
Forks, patches and other feedback are welcome.
Nacho López @mrmans0n
The MIT License (MIT)
Copyright (c) 2013-2017 Nacho Lopez
Permission is hereby granted, free of charge, to any person obtaining a copyof this software and associated documentation files (the "Software"), to dealin the Software without restriction, including without limitation the rightsto use, copy, modify, merge, publish, distribute, sublicense, and/or sellcopies of the Software, and to permit persons to whom the Software isfurnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included inall copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS ORIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THEAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHERLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS INTHE SOFTWARE.
git服务器搭建过程 A.首先要选择搭建git时的协议,在文档里有三种协议 1.本地协议:就是多人使用一台电脑,就是使用不同的账号登录同一台电脑,使用共享文件来同时进行操作,很危险。使用命令为: git clone /opt/git/project.git 2.HTTP 协议: http协议又分为两种 一种为智能(Smart) HTTP 协议 和 哑(Dumb) HTT
Smart HTTP是通过激活Git(1.6.6及以后版本)自带的、名为git-http-backend的cgi脚本实现的。注意,此脚本只在Apache 2.x上有效。 配置Apache2,开启git-http-backend脚本: SetEnv GIT_PROJECT_ROOT /my_repos_root_path SetEnv GIT_HTTP_EXPORT_ALL ScriptAlias
下表描述了位置功能: 功能 描述 maxloc(array, mask) 它返回数组数组中最大元素的位置,如果仅包含掩码的掩码,则返回位置,返回位置,结果为整数向量。 minloc(array, mask) 它返回数组数组中最小元素的位置,如果仅包含掩码的掩码,则返回位置,返回位置,结果为整数向量。 Example 以下示例演示了以下概念: program arrayLocation implic
Location响应头指示URL的页面重定向到。它仅在提供3xx状态响应时才提供意义。 用于使新请求获取指向的页面的 HTTP 方法Location取决于原始方法和重定向类型: 如果303(另请参见)响应始终导致使用GET方法,307(临时重定向)和308(永久重定向)不会更改原始请求中使用的方法; 301(永久重定向)和302(找到)不会改变大部分时间的方法,虽然较老的用户代理可能(所以你基本上
我们一般通过 SSH 进行授权访问,通过 git:// 进行无授权访问,但是还有一种协议可以同时实现以上两种方式的访问。 设置 Smart HTTP 一般只需要在服务器上启用一个 Git 自带的名为 git-http-backend 的 CGI 脚本。 该 CGI 脚本将会读取由 git fetch 或 git push 命令向 HTTP URL 发送的请求路径和头部信息,来判断该客户端是否支持
Smart Select 可以自动帮你把原生的select变成一个由 分组单选按钮 构成的动态页面。在很多iOs native应用中你可以看到这种特性。 Smart Select 布局 Smart Select 布局非常简单,只需要在 列表 中插入 <select>,并且给 item-link 加上 "smart-select" class即可。 <div class="list-block">
无论是评估您的每日健康水平、制定个人训练计划、在合适的强度下训练,还是接收即时反馈,Smart Coaching 均可提供独特、易用的功能来满足您的个人需求,并让您在训练时享受最大乐趣,并发挥出最大的积极性。 手表的 Polar Smart Coaching 具有以下功能: Training Load Pro FuelWise™ 手腕式跑步功率测量 跑步计划 跑步指数 训练效益 智能卡路里 持续心
无论是评估您的每日健康水平、制定个人训练计划、在合适的强度下训练,还是接收即时反馈,Smart Coaching 均可提供独特、易用的功能来满足您的个人需求,并让您在训练时享受最大乐趣,并发挥出最大的积极性。 手表的 Polar Smart Coaching 具有以下功能: Training Load Pro FuelWise™ 手腕式跑步功率测量 跑步计划 跑步指数 训练效益 智能卡路里 持续心