// License: GPL. For details, see LICENSE file.
import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.OutputStream;
import java.io.StringWriter;
import java.io.UncheckedIOException;
import java.io.Writer;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.time.Instant;
import java.time.ZoneId;
import java.time.format.DateTimeFormatter;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.EnumSet;
import java.util.List;
import java.util.Locale;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import javax.imageio.ImageIO;
import javax.json.Json;
import javax.json.JsonArrayBuilder;
import javax.json.JsonObjectBuilder;
import javax.json.JsonWriter;
import javax.json.stream.JsonGenerator;
import org.openstreetmap.josm.actions.DeleteAction;
import org.openstreetmap.josm.command.DeleteCommand;
import org.openstreetmap.josm.data.Preferences;
import org.openstreetmap.josm.data.Version;
import org.openstreetmap.josm.data.coor.LatLon;
import org.openstreetmap.josm.data.osm.Node;
import org.openstreetmap.josm.data.osm.OsmPrimitive;
import org.openstreetmap.josm.data.osm.Tag;
import org.openstreetmap.josm.data.osm.Way;
import org.openstreetmap.josm.data.osm.visitor.paint.MapPaintSettings;
import org.openstreetmap.josm.data.osm.visitor.paint.StyledMapRenderer;
import org.openstreetmap.josm.data.preferences.JosmBaseDirectories;
import org.openstreetmap.josm.data.preferences.JosmUrls;
import org.openstreetmap.josm.data.preferences.sources.ExtendedSourceEntry;
import org.openstreetmap.josm.data.preferences.sources.SourceEntry;
import org.openstreetmap.josm.data.projection.ProjectionRegistry;
import org.openstreetmap.josm.data.projection.Projections;
import org.openstreetmap.josm.gui.NavigatableComponent;
import org.openstreetmap.josm.gui.mappaint.Cascade;
import org.openstreetmap.josm.gui.mappaint.Environment;
import org.openstreetmap.josm.gui.mappaint.MapPaintStyles;
import org.openstreetmap.josm.gui.mappaint.MultiCascade;
import org.openstreetmap.josm.gui.mappaint.mapcss.ConditionFactory;
import org.openstreetmap.josm.gui.mappaint.mapcss.MapCSSRule;
import org.openstreetmap.josm.gui.mappaint.mapcss.MapCSSStyleSource;
import org.openstreetmap.josm.gui.mappaint.mapcss.Selector;
import org.openstreetmap.josm.gui.mappaint.mapcss.parsergen.MapCSSParser;
import org.openstreetmap.josm.gui.mappaint.mapcss.parsergen.ParseException;
import org.openstreetmap.josm.gui.mappaint.styleelement.AreaElement;
import org.openstreetmap.josm.gui.mappaint.styleelement.LineElement;
import org.openstreetmap.josm.gui.mappaint.styleelement.StyleElement;
import org.openstreetmap.josm.gui.preferences.map.TaggingPresetPreference;
import org.openstreetmap.josm.gui.tagging.presets.TaggingPreset;
import org.openstreetmap.josm.gui.tagging.presets.TaggingPresetReader;
import org.openstreetmap.josm.gui.tagging.presets.TaggingPresetType;
import org.openstreetmap.josm.gui.tagging.presets.items.KeyedItem;
import org.openstreetmap.josm.io.CachedFile;
import org.openstreetmap.josm.io.OsmTransferException;
import org.openstreetmap.josm.spi.preferences.Config;
import org.openstreetmap.josm.tools.Logging;
import org.openstreetmap.josm.tools.OptionParser;
import org.openstreetmap.josm.tools.RightAndLefthandTraffic;
import org.openstreetmap.josm.tools.Territories;
import org.openstreetmap.josm.tools.Utils;
import org.xml.sax.SAXException;
/**
* Extracts tag information for the taginfo project.
*
* Run from the base directory of a JOSM checkout:
*
* java -cp dist/josm-custom.jar TagInfoExtract --type mappaint
* java -cp dist/josm-custom.jar TagInfoExtract --type presets
* java -cp dist/josm-custom.jar TagInfoExtract --type external_presets
*/
public class TagInfoExtract {
/**
* Main method.
*/
public static void main(String[] args) throws Exception {
TagInfoExtract script = new TagInfoExtract();
script.parseCommandLineArguments(args);
script.init();
switch (script.options.mode) {
case MAPPAINT:
script.new StyleSheet().run();
break;
case PRESETS:
script.new Presets().run();
break;
case EXTERNAL_PRESETS:
script.new ExternalPresets().run();
break;
default:
throw new IllegalStateException("Invalid type " + script.options.mode);
}
if (!script.options.noexit) {
System.exit(0);
}
}
enum Mode {
MAPPAINT, PRESETS, EXTERNAL_PRESETS
}
private final Options options = new Options();
/**
* Parse command line arguments.
*/
private void parseCommandLineArguments(String[] args) {
if (args.length == 1 && "--help".equals(args[0])) {
this.usage();
}
final OptionParser parser = new OptionParser(getClass().getName());
parser.addArgumentParameter("type", OptionParser.OptionCount.REQUIRED, options::setMode);
parser.addArgumentParameter("input", OptionParser.OptionCount.OPTIONAL, options::setInputFile);
parser.addArgumentParameter("output", OptionParser.OptionCount.OPTIONAL, options::setOutputFile);
parser.addArgumentParameter("imgdir", OptionParser.OptionCount.OPTIONAL, options::setImageDir);
parser.addArgumentParameter("imgurlprefix", OptionParser.OptionCount.OPTIONAL, options::setImageUrlPrefix);
parser.addFlagParameter("noexit", options::setNoExit);
parser.addFlagParameter("help", this::usage);
parser.parseOptionsOrExit(Arrays.asList(args));
}
private void usage() {
System.out.println("java " + getClass().getName());
System.out.println(" --type TYPE\tthe project type to be generated: " + Arrays.toString(Mode.values()));
System.out.println(" --input FILE\tthe input file to use (overrides defaults for types mappaint, presets)");
System.out.println(" --output FILE\tthe output file to use (defaults to STDOUT)");
System.out.println(" --imgdir DIRECTORY\tthe directory to put the generated images in (default: " + options.imageDir + ")");
System.out.println(" --imgurlprefix STRING\timage URLs prefix for generated image files (public path on webserver)");
System.out.println(" --noexit\tdo not call System.exit(), for use from Ant script");
System.out.println(" --help\tshow this help");
System.exit(0);
}
private static class Options {
Mode mode;
int josmSvnRevision = Version.getInstance().getVersion();
Path baseDir = Paths.get("");
Path imageDir = Paths.get("taginfo-img");
String imageUrlPrefix;
CachedFile inputFile;
Path outputFile;
boolean noexit;
void setMode(String value) {
mode = Mode.valueOf(value.toUpperCase(Locale.ENGLISH));
switch (mode) {
case MAPPAINT:
inputFile = new CachedFile("resource://styles/standard/elemstyles.mapcss");
break;
case PRESETS:
inputFile = new CachedFile("resource://data/defaultpresets.xml");
break;
default:
inputFile = null;
}
}
void setInputFile(String value) {
inputFile = new CachedFile(value);
}
void setOutputFile(String value) {
outputFile = Paths.get(value);
}
void setImageDir(String value) {
imageDir = Paths.get(value);
}
void setImageUrlPrefix(String value) {
imageUrlPrefix = value;
}
void setNoExit() {
noexit = true;
}
/**
* Determine full image url (can refer to JOSM or OSM repository).
* @param path the image path
*/
private String findImageUrl(String path) {
final Path f = baseDir.resolve("images").resolve(path);
if (Files.exists(f)) {
return "https://josm.openstreetmap.de/export/" + josmSvnRevision + "/josm/trunk/images/" + path;
}
throw new IllegalStateException("Cannot find image url for " + path);
}
}
private abstract class Extractor {
abstract void run() throws Exception;
void writeJson(String name, String description, Iterable tags) throws IOException {
try (Writer writer = options.outputFile != null ? Files.newBufferedWriter(options.outputFile) : new StringWriter();
JsonWriter json = Json
.createWriterFactory(Collections.singletonMap(JsonGenerator.PRETTY_PRINTING, true))
.createWriter(writer)) {
JsonObjectBuilder project = Json.createObjectBuilder()
.add("name", name)
.add("description", description)
.add("project_url", "https://josm.openstreetmap.de/")
.add("icon_url", "https://josm.openstreetmap.de/export/7770/josm/trunk/images/logo_16x16x8.png")
.add("contact_name", "JOSM developer team")
.add("contact_email", "josm-dev@openstreetmap.org");
final JsonArrayBuilder jsonTags = Json.createArrayBuilder();
for (TagInfoTag t : tags) {
jsonTags.add(t.toJson());
}
json.writeObject(Json.createObjectBuilder()
.add("data_format", 1)
.add("data_updated", DateTimeFormatter.ofPattern("yyyyMMdd'T'hhmmss'Z'").withZone(ZoneId.of("Z")).format(Instant.now()))
.add("project", project)
.add("tags", jsonTags)
.build());
if (options.outputFile == null) {
System.out.println(writer.toString());
}
}
}
}
private class Presets extends Extractor {
@Override
void run() throws IOException, OsmTransferException, SAXException {
try (BufferedReader reader = options.inputFile.getContentReader()) {
Collection presets = TaggingPresetReader.readAll(reader, true);
List tags = convertPresets(presets, "", true);
writeJson("JOSM main presets", "Tags supported by the default presets in the OSM editor JOSM", tags);
}
}
List convertPresets(Iterable presets, String descriptionPrefix, boolean addImages) {
final List tags = new ArrayList<>();
for (TaggingPreset preset : presets) {
for (KeyedItem item : Utils.filteredCollection(preset.data, KeyedItem.class)) {
final Iterable values = item.isKeyRequired()
? item.getValues()
: Collections.emptyList();
for (String value : values) {
final Set types = preset.types == null ? Collections.emptySet() : preset.types.stream()
.map(it -> TaggingPresetType.CLOSEDWAY.equals(it)
? TagInfoTag.Type.AREA
: TaggingPresetType.MULTIPOLYGON.equals(it)
? TagInfoTag.Type.RELATION
: TagInfoTag.Type.valueOf(it.toString()))
.collect(Collectors.toCollection(() -> EnumSet.noneOf(TagInfoTag.Type.class)));
tags.add(new TagInfoTag(descriptionPrefix + preset.getName(), item.key, value, types,
addImages && preset.iconName != null ? options.findImageUrl(preset.iconName) : null));
}
}
}
return tags;
}
}
private class ExternalPresets extends Presets {
@Override
void run() throws IOException, OsmTransferException, SAXException {
TaggingPresetReader.setLoadIcons(false);
final Collection sources = new TaggingPresetPreference.TaggingPresetSourceEditor().loadAndGetAvailableSources();
final List tags = new ArrayList<>();
for (SourceEntry source : sources) {
if (source.url.startsWith("resource")) {
// default presets
continue;
}
try {
System.out.println("Loading " + source.url);
Collection presets = TaggingPresetReader.readAll(source.url, false);
final List t = convertPresets(presets, source.title + " ", false);
System.out.println("Converting " + t.size() + " presets of " + source.title);
tags.addAll(t);
} catch (Exception ex) {
System.err.println("Skipping " + source.url + " due to error");
ex.printStackTrace();
}
}
writeJson("JOSM user presets", "Tags supported by the user contributed presets in the OSM editor JOSM", tags);
}
}
private class StyleSheet extends Extractor {
private MapCSSStyleSource styleSource;
@Override
void run() throws IOException, ParseException {
init();
parseStyleSheet();
final List tags = convertStyleSheet();
writeJson("JOSM main mappaint style", "Tags supported by the main mappaint style in the OSM editor JOSM", tags);
}
/**
* Read the style sheet file and parse the MapCSS code.
*/
private void parseStyleSheet() throws IOException, ParseException {
try (BufferedReader reader = options.inputFile.getContentReader()) {
MapCSSParser parser = new MapCSSParser(reader, MapCSSParser.LexicalState.DEFAULT);
styleSource = new MapCSSStyleSource("");
styleSource.url = "";
parser.sheet(styleSource);
}
}
/**
* Collect all the tag from the style sheet.
*/
private List convertStyleSheet() {
return styleSource.rules.stream()
.map(rule -> rule.selector)
.filter(Selector.GeneralSelector.class::isInstance)
.map(Selector.GeneralSelector.class::cast)
.map(Selector.AbstractSelector::getConditions)
.flatMap(Collection::stream)
.filter(ConditionFactory.SimpleKeyValueCondition.class::isInstance)
.map(ConditionFactory.SimpleKeyValueCondition.class::cast)
.map(condition -> condition.asTag(null))
.distinct()
.map(tag -> {
String iconUrl = null;
final EnumSet types = EnumSet.noneOf(TagInfoTag.Type.class);
Optional nodeUrl = new NodeChecker(tag).findUrl(true);
if (nodeUrl.isPresent()) {
iconUrl = nodeUrl.get();
types.add(TagInfoTag.Type.NODE);
}
Optional wayUrl = new WayChecker(tag).findUrl(iconUrl == null);
if (wayUrl.isPresent()) {
if (iconUrl == null) {
iconUrl = wayUrl.get();
}
types.add(TagInfoTag.Type.WAY);
}
Optional areaUrl = new AreaChecker(tag).findUrl(iconUrl == null);
if (areaUrl.isPresent()) {
if (iconUrl == null) {
iconUrl = areaUrl.get();
}
types.add(TagInfoTag.Type.AREA);
}
return new TagInfoTag(null, tag.getKey(), tag.getValue(), types, iconUrl);
})
.collect(Collectors.toList());
}
/**
* Check if a certain tag is supported by the style as node / way / area.
*/
private abstract class Checker {
Checker(Tag tag) {
this.tag = tag;
}
Environment applyStylesheet(OsmPrimitive osm) {
osm.put(tag);
MultiCascade mc = new MultiCascade();
Environment env = new Environment(osm, mc, null, styleSource);
for (MapCSSRule r : styleSource.rules) {
env.clearSelectorMatchingInformation();
if (r.selector.matches(env)) {
// ignore selector range
if (env.layer == null) {
env.layer = "default";
}
r.execute(env);
}
}
env.layer = "default";
return env;
}
/**
* Create image file from StyleElement.
*
* @return the URL
*/
String createImage(StyleElement element, final String type, NavigatableComponent nc) {
BufferedImage img = new BufferedImage(16, 16, BufferedImage.TYPE_INT_ARGB);
Graphics2D g = img.createGraphics();
g.setClip(0, 0, 16, 16);
StyledMapRenderer renderer = new StyledMapRenderer(g, nc, false);
renderer.getSettings(false);
element.paintPrimitive(osm, MapPaintSettings.INSTANCE, renderer, false, false, false);
final String imageName = type + "_" + tag + ".png";
try (OutputStream out = Files.newOutputStream(options.imageDir.resolve(imageName))) {
ImageIO.write(img, "png", out);
} catch (IOException e) {
throw new UncheckedIOException(e);
}
final String baseUrl = options.imageUrlPrefix != null ? options.imageUrlPrefix : options.imageDir.toString();
return baseUrl + "/" + imageName;
}
/**
* Checks, if tag is supported and find URL for image icon in this case.
*
* @param generateImage if true, create or find a suitable image icon and return URL,
* if false, just check if tag is supported and return true or false
*/
abstract Optional findUrl(boolean generateImage);
protected Tag tag;
protected OsmPrimitive osm;
}
private class NodeChecker extends Checker {
NodeChecker(Tag tag) {
super(tag);
}
@Override
Optional findUrl(boolean generateImage) {
this.osm = new Node(LatLon.ZERO);
Environment env = applyStylesheet(osm);
Cascade c = env.mc.getCascade("default");
Object image = c.get("icon-image");
if (image instanceof MapPaintStyles.IconReference && !((MapPaintStyles.IconReference) image).isDeprecatedIcon()) {
return Optional.of(options.findImageUrl(((MapPaintStyles.IconReference) image).iconName));
}
return Optional.empty();
}
}
private class WayChecker extends Checker {
WayChecker(Tag tag) {
super(tag);
}
@Override
Optional findUrl(boolean generateImage) {
this.osm = new Way();
NavigatableComponent nc = new NavigatableComponent();
Node n1 = new Node(nc.getLatLon(2, 8));
Node n2 = new Node(nc.getLatLon(14, 8));
((Way) osm).addNode(n1);
((Way) osm).addNode(n2);
Environment env = applyStylesheet(osm);
LineElement les = LineElement.createLine(env);
if (les != null) {
if (!generateImage) return Optional.of("");
return Optional.of(createImage(les, "way", nc));
}
return Optional.empty();
}
}
private class AreaChecker extends Checker {
AreaChecker(Tag tag) {
super(tag);
}
@Override
Optional findUrl(boolean generateImage) {
this.osm = new Way();
NavigatableComponent nc = new NavigatableComponent();
Node n1 = new Node(nc.getLatLon(2, 2));
Node n2 = new Node(nc.getLatLon(14, 2));
Node n3 = new Node(nc.getLatLon(14, 14));
Node n4 = new Node(nc.getLatLon(2, 14));
((Way) osm).addNode(n1);
((Way) osm).addNode(n2);
((Way) osm).addNode(n3);
((Way) osm).addNode(n4);
((Way) osm).addNode(n1);
Environment env = applyStylesheet(osm);
AreaElement aes = AreaElement.create(env);
if (aes != null) {
if (!generateImage) return Optional.of("");
return Optional.of(createImage(aes, "area", nc));
}
return Optional.empty();
}
}
}
/**
* POJO representing a Taginfo tag.
*/
private static class TagInfoTag {
final String description;
final String key;
final String value;
final Set objectTypes;
final String iconURL;
TagInfoTag(String description, String key, String value, Set objectTypes, String iconURL) {
this.description = description;
this.key = key;
this.value = value;
this.objectTypes = objectTypes;
this.iconURL = iconURL;
}
JsonObjectBuilder toJson() {
final JsonObjectBuilder object = Json.createObjectBuilder();
if (description != null) {
object.add("description", description);
}
object.add("key", key);
object.add("value", value);
if ((!objectTypes.isEmpty())) {
final JsonArrayBuilder types = Json.createArrayBuilder();
objectTypes.stream().map(Enum::name).map(String::toLowerCase).forEach(types::add);
object.add("object_types", types);
}
if (iconURL != null) {
object.add("icon_url", iconURL);
}
return object;
}
enum Type {
NODE, WAY, AREA, RELATION
}
}
/**
* Initialize the script.
*/
private void init() throws IOException {
Logging.setLogLevel(Logging.LEVEL_INFO);
Preferences.main().enableSaveOnPut(false);
Config.setPreferencesInstance(Preferences.main());
Config.setBaseDirectoriesProvider(JosmBaseDirectories.getInstance());
Config.setUrlsProvider(JosmUrls.getInstance());
ProjectionRegistry.setProjection(Projections.getProjectionByCode("EPSG:3857"));
Path tmpdir = Files.createTempDirectory(options.baseDir, "pref");
tmpdir.toFile().deleteOnExit();
System.setProperty("josm.home", tmpdir.toString());
DeleteCommand.setDeletionCallback(DeleteAction.defaultDeletionCallback);
Territories.initialize();
RightAndLefthandTraffic.initialize();
Files.createDirectories(options.imageDir);
}
}