вы передаете значение 8f
(с плавающей запятой) как radius
там, тогда как radius
должно быть 8.0d
или Double.valueOf(8.0)
, где MAX_SUPPORTED_RADIUS
равно 8587
километрам.
пока проблема в том, что GeoFire
уже нужно знать о .child("location")
, но это невозможно представить с помощью Reference
; только DataSnapshot
имеет getChildren()
.
нижняя строка:
вам нужно будет создать отдельные местоположения Reference
, чтобы избежать вложения. тем не менее, вы все равно можете использовать связанный ключ uid
для этих узлов (или, по крайней мере, добавить его как дочерний узел), чтобы иметь возможность искать в users Reference
. это отношение 1:1
между двумя Reference
с.
так вот рабочий Java
пример, просто потому что ...
мы предполагаем следующую структуру (как описано выше):
{
"locations" : {
"CR88aa9gnDfJYYGq5ZTMwwC38C12" : {
".priority" : "9q8yywdgue",
"g" : "9q8yywdgue",
"l" : [ 37.7853889, -122.4056973 ]
}
},
"users" : {
"CR88aa9gnDfJYYGq5ZTMwwC38C12" : {
"displayName" : "user 01",
...
}
}
}
правила базы данных должны иметь .indexOn
для locations
поле g
установлено:
{
"rules": {
...
"locations": {
".indexOn": "g"
}
}
}
зависимости в модуле build.gradle
:
dependencies {
...
implementation "com.firebase:geofire-android:2.3.1"
}
и это демонстрирует, как получить снимок пользователя по результату GeoQuery
;
обратите внимание на GeoQueryEventListener
вместо GeoQueryDataEventListener
:
public class GeofireActivity extends AppCompatActivity {
private static final String LOG_TAG = GeofireActivity.class.getSimpleName();
private DatabaseReference refBase = null;
private DatabaseReference refLocation = null;
private DatabaseReference refUser = null;
private GeoFire geoFire = null;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
this.setContentView(R.layout.fragment_geofire);
this.setReferences();
}
private void setReferences() {
this.refBase = FirebaseDatabase.getInstance().getReference();
this.refUser = refBase.child("users");
this.refLocation = refBase.child("locations");
this.geoFire = new GeoFire(this.refLocation);
}
private void searchNearby(double latitude, double longitude, double radius) {
this.searchNearby(new GeoLocation(latitude, longitude), radius);
}
private void searchNearby(GeoLocation location, double radius) {
GeoQuery geoQuery = this.geoFire.queryAtLocation(location, radius);
geoQuery.addGeoQueryEventListener(new GeoQueryEventListener() {
@Override
public void onKeyEntered(String key, GeoLocation location) {
String loc = String.valueOf(location.latitude) + ", " + String.valueOf(location.longitude);
Log.d(LOG_TAG, "onKeyEntered: " + key + " @ " + loc);
/* once the key is known, one can lookup the associated record */
refUser.child(key).addListenerForSingleValueEvent(new ValueEventListener() {
@Override
public void onDataChange(@NonNull DataSnapshot dataSnapshot) {
Log.d(LOG_TAG, "onDataChange: " + dataSnapshot.toString());
}
@Override
public void onCancelled(@NonNull DatabaseError firebaseError) {
Log.e(LOG_TAG, "onCancelled: " + firebaseError.getMessage());
}
});
}
@Override
public void onKeyExited(String key) {
Log.d(LOG_TAG, "onKeyExited: " + key);
}
@Override
public void onKeyMoved(String key, GeoLocation location) {
Log.d(LOG_TAG, "onKeyMoved: " + key);
}
@Override
public void onGeoQueryReady() {
Log.d(LOG_TAG, "onGeoQueryReady");
}
@Override
public void onGeoQueryError(DatabaseError error) {
Log.e(LOG_TAG, "onGeoQueryError" + error.getMessage());
}
});
}
}
Чтобы сохранить целостность, при удалении пользовательской записи потребуется удалить связанную запись о местоположении, иначе это приведет к ключам, которые больше не могут быть найдены.