Abstract style model further from MapLibre

This commit is contained in:
2026-04-16 17:25:14 +02:00
parent cb892d1bbd
commit ce4ce2f2ac
13 changed files with 121 additions and 59 deletions

1
.gitignore vendored
View File

@@ -9,3 +9,4 @@ DS_Store
.cxx .cxx
local.properties local.properties
/legacy_code/ /legacy_code/
maptiler*

View File

@@ -5,6 +5,7 @@
"sources": { "sources": {
"satelite": { "satelite": {
"type": "raster", "type": "raster",
"schema" : "raster",
"tiles": [ "tiles": [
"https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}" "https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}"
], ],
@@ -14,8 +15,6 @@
"tileSize": 512 "tileSize": 512
} }
}, },
"sprite": "",
"glyphs": "https://orangemug.github.io/font-glyphs/glyphs/{fontstack}/{range}.pbf",
"layers": [ "layers": [
{ {
"id": "satelite", "id": "satelite",

View File

@@ -1,16 +1,11 @@
package eu.konggdev.strikemaps.map.layer; package eu.konggdev.strikemaps.map.layer;
import org.maplibre.android.style.layers.Layer; import com.fasterxml.jackson.databind.JsonNode;
import org.maplibre.android.style.sources.GeoJsonSource;
//FIXME: Get rid of reliance on MapLibre!
//Most likely implement an "AdditionalMapLayer" or something of that sorts (?)
public class MapLayer { public class MapLayer {
public GeoJsonSource source; public JsonNode layer;
public Layer layer;
public MapLayer(GeoJsonSource source, Layer layer) { public MapLayer(JsonNode layer) {
this.source = source;
this.layer = layer; this.layer = layer;
} }
} }

View File

@@ -0,0 +1,16 @@
package eu.konggdev.strikemaps.map.layer;
import com.fasterxml.jackson.databind.JsonNode;
import eu.konggdev.strikemaps.map.source.MapSource;
public class SourcedMapLayer {
public String key;
public MapSource source;
public JsonNode layer;
public SourcedMapLayer(String key, MapSource source, JsonNode layer) {
this.key = key;
this.source = source;
this.layer = layer;
}
}

View File

@@ -1,8 +1,8 @@
package eu.konggdev.strikemaps.map.overlay; package eu.konggdev.strikemaps.map.overlay;
import eu.konggdev.strikemaps.map.layer.MapLayer; import eu.konggdev.strikemaps.map.layer.SourcedMapLayer;
/* More or less a data-driven layer factory */ /* More or less a data-driven layer factory */
public interface MapOverlay { public interface MapOverlay {
public MapLayer makeLayer(); public SourcedMapLayer makeLayer();
} }

View File

@@ -6,18 +6,18 @@ import android.location.LocationListener;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import eu.konggdev.strikemaps.app.AppController; import eu.konggdev.strikemaps.app.AppController;
import eu.konggdev.strikemaps.map.MapComponent; import eu.konggdev.strikemaps.map.MapComponent;
import eu.konggdev.strikemaps.map.layer.MapLayer; import eu.konggdev.strikemaps.map.layer.SourcedMapLayer;
import eu.konggdev.strikemaps.map.overlay.MapOverlay; import eu.konggdev.strikemaps.map.overlay.MapOverlay;
import eu.konggdev.strikemaps.map.source.MapSource;
import eu.konggdev.strikemaps.data.provider.LocationDataProvider; import eu.konggdev.strikemaps.data.provider.LocationDataProvider;
import org.maplibre.android.style.layers.CircleLayer;
import org.maplibre.android.style.layers.Property;
import org.maplibre.android.style.sources.GeoJsonSource;
import org.maplibre.geojson.Feature; import org.maplibre.geojson.Feature;
import org.maplibre.geojson.FeatureCollection; import org.maplibre.geojson.FeatureCollection;
import org.maplibre.geojson.Point; import org.maplibre.geojson.Point;
import static org.maplibre.android.style.layers.PropertyFactory.*; import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
public class LocationOverlay implements MapOverlay, LocationListener { public class LocationOverlay implements MapOverlay, LocationListener {
LocationDataProvider locationDataProvider; LocationDataProvider locationDataProvider;
@@ -33,25 +33,55 @@ public class LocationOverlay implements MapOverlay, LocationListener {
} }
@Override @Override
public MapLayer makeLayer() { public SourcedMapLayer makeLayer() {
GeoJsonSource source = new GeoJsonSource( MapSource source = new MapSource();
"location",
FeatureCollection.fromFeatures(new Feature[]{}) // empty
);
if (currentLocation != null) source.type = "geojson";
source.setGeoJson(Feature.fromGeometry(Point.fromLngLat(currentLocation.getLongitude(), currentLocation.getLatitude())));
CircleLayer layer = new CircleLayer("location", "location"); ObjectMapper mapper = new ObjectMapper();
layer.setProperties( try {
circleRadius(5f), ObjectNode data = mapper.createObjectNode();
circleColor(Color.parseColor("#1E88E5")), data.put("type", "Feature");
circleStrokeColor(Color.WHITE),
circleStrokeWidth(1.5f),
circlePitchAlignment(Property.CIRCLE_PITCH_ALIGNMENT_MAP)
);
return new MapLayer(source, layer); if(currentLocation != null) {
ObjectNode geometry = mapper.createObjectNode();
geometry.put("type", "Point");
ArrayNode coordinates = mapper.createArrayNode();
coordinates.add(currentLocation.getLongitude());
coordinates.add(currentLocation.getLatitude());
geometry.set("coordinates", coordinates);
data.set("geometry", geometry);
data.set("properties", mapper.createObjectNode());
}
source.data = data;
} catch (Exception e) {
e.printStackTrace();
}
ObjectNode layer = mapper.createObjectNode();
layer.put("id", "location");
layer.put("type", "circle");
layer.put("source", "location");
ObjectNode paint = mapper.createObjectNode();
paint.put("circle-radius", 5);
paint.put("circle-color", "#1E88E5");
paint.put("circle-stroke-color", "#FFFFFF");
paint.put("circle-stroke-width", 1.5);
layer.set("paint", paint);
ObjectNode layout = mapper.createObjectNode();
layout.put("circle-pitch-alignment", "map");
layer.set("layout", layout);
ArrayNode layers = mapper.createArrayNode();
layers.add(layer);
return new SourcedMapLayer("location", source, layers);
} }
@Override @Override

View File

@@ -1,11 +1,12 @@
package eu.konggdev.strikemaps.map.overlay.implementation; package eu.konggdev.strikemaps.map.overlay.implementation;
import eu.konggdev.strikemaps.map.layer.MapLayer; import eu.konggdev.strikemaps.map.layer.MapLayer;
import eu.konggdev.strikemaps.map.layer.SourcedMapLayer;
import eu.konggdev.strikemaps.map.overlay.MapOverlay; import eu.konggdev.strikemaps.map.overlay.MapOverlay;
public class PointSelectionOverlay implements MapOverlay { public class PointSelectionOverlay implements MapOverlay {
@Override @Override
public MapLayer makeLayer() { public SourcedMapLayer makeLayer() {
return null; return null;
} }
} }

View File

@@ -15,5 +15,6 @@ public interface MapRenderer {
View getView(); View getView();
//TODO: Get rid of MapLibre Feature class dependence
List<Feature> featuresAtPoint(LatLng point); List<Feature> featuresAtPoint(LatLng point);
} }

View File

@@ -4,11 +4,13 @@ import android.view.View;
import androidx.annotation.NonNull; import androidx.annotation.NonNull;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ObjectNode; import com.fasterxml.jackson.databind.node.ObjectNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import eu.konggdev.strikemaps.data.helper.UserPrefsHelper; import eu.konggdev.strikemaps.data.helper.UserPrefsHelper;
import eu.konggdev.strikemaps.map.overlay.MapOverlay; import eu.konggdev.strikemaps.map.overlay.MapOverlay;
import eu.konggdev.strikemaps.map.layer.MapLayer; import eu.konggdev.strikemaps.map.layer.SourcedMapLayer;
import eu.konggdev.strikemaps.map.renderer.MapRenderer; import eu.konggdev.strikemaps.map.renderer.MapRenderer;
import eu.konggdev.strikemaps.map.style.MapStyle; import eu.konggdev.strikemaps.map.style.MapStyle;
import org.maplibre.android.MapLibre; import org.maplibre.android.MapLibre;
@@ -39,25 +41,38 @@ public class MapLibreNativeRenderer implements MapRenderer, OnMapReadyCallback {
mapView.getMapAsync(this); mapView.getMapAsync(this);
} }
void passLayer(MapLayer layer) {
map.getStyle().addSource(layer.source);
map.getStyle().addLayer(layer.layer);
}
@Override @Override
public void reload() { public void reload() {
ObjectMapper mapper = new ObjectMapper(); ObjectMapper mapper = new ObjectMapper();
mapper.setSerializationInclusion(JsonInclude.Include.NON_EMPTY);
MapStyle style = controller.style; MapStyle style = controller.style;
try { try {
/* Take metadata from MapStyle
everything outside sources, layers */
ObjectNode root = style.metadata.deepCopy(); ObjectNode root = style.metadata.deepCopy();
root.set("sources", mapper.valueToTree(style.sources));
root.set("layers", style.layerDefinitions); //Sources
map.setStyle(new Style.Builder().fromJson(mapper.writeValueAsString(root)), intStyle -> { ObjectNode sources = mapper.createObjectNode();
for(MapOverlay overlay : controller.overlays.values()) { style.sources.forEach((k, v) -> sources.set(k, mapper.valueToTree(v)));
passLayer(overlay.makeLayer());
//Layers
ArrayNode layers = mapper.createArrayNode();
layers.addAll((ArrayNode) style.layerDefinitions);
//Overlays
for (MapOverlay overlay : controller.overlays.values()) {
SourcedMapLayer overlayLayer = overlay.makeLayer();
sources.set(overlayLayer.key, mapper.valueToTree(overlayLayer.source));
layers.addAll((ArrayNode) overlayLayer.layer);
} }
});
//Set all to root
root.set("sources", sources);
root.set("layers", layers);
map.setStyle(new Style.Builder().fromJson(mapper.writeValueAsString(root)));
} catch (Exception e) { } catch (Exception e) {
app.logcat("Failed to reload Map");
e.printStackTrace(); e.printStackTrace();
} }
} }

View File

@@ -4,15 +4,18 @@ import com.fasterxml.jackson.databind.JsonNode;
public class MapSource { public class MapSource {
public String url; public String url;
public JsonNode data;
public String type; public String type;
public String schema; public String schema;
public String attribution;
/* For raster sources */
public JsonNode tiles; public JsonNode tiles;
public int minzoom; public int minzoom;
public int maxzoom; public int maxzoom;
public String scheme;
public int tileSize;
public String attribution;
public String encoding;
public MapSource() { } public MapSource() { }
} }

View File

@@ -6,6 +6,7 @@ import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.JsonNode; import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ObjectNode; import com.fasterxml.jackson.databind.node.ObjectNode;
import com.fasterxml.jackson.databind.node.ArrayNode;
import eu.konggdev.strikemaps.app.AppController; import eu.konggdev.strikemaps.app.AppController;
import eu.konggdev.strikemaps.data.helper.FileHelper; import eu.konggdev.strikemaps.data.helper.FileHelper;
import eu.konggdev.strikemaps.map.source.MapSource; import eu.konggdev.strikemaps.map.source.MapSource;
@@ -19,7 +20,7 @@ public class MapStyle {
public JsonNode metadata; // everything except layers + sources public JsonNode metadata; // everything except layers + sources
public Map<String, MapSource> sources; public Map<String, MapSource> sources;
public JsonNode layerDefinitions; // the "layers" array public ArrayNode layerDefinitions; // the "layers" array
//FIXME //FIXME
public static MapStyle fromMapLibreJsonFile(String filename, AppController app) { public static MapStyle fromMapLibreJsonFile(String filename, AppController app) {
@@ -40,7 +41,7 @@ public class MapStyle {
new TypeReference<Map<String, MapSource>>() {} new TypeReference<Map<String, MapSource>>() {}
); );
style.layerDefinitions = root.path("layers"); style.layerDefinitions = root.withArray("layers");
ObjectNode metadata = root.deepCopy(); ObjectNode metadata = root.deepCopy();
metadata.remove("layers"); metadata.remove("layers");