// License: GPL. For details, see LICENSE file.
package org.openstreetmap.josm.plugins.utilsplugin2.latlon;
import static org.openstreetmap.josm.tools.I18n.tr;
import java.awt.Color;
import java.awt.Component;
import java.awt.GridBagLayout;
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;
import java.text.NumberFormat;
import java.text.ParsePosition;
import java.util.ArrayList;
import java.util.List;
import java.util.Locale;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.swing.BorderFactory;
import javax.swing.ButtonGroup;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JRadioButton;
import javax.swing.JScrollPane;
import javax.swing.JSeparator;
import javax.swing.JTabbedPane;
import javax.swing.JTextArea;
import javax.swing.UIManager;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import org.openstreetmap.josm.Main;
import org.openstreetmap.josm.data.coor.CoordinateFormat;
import org.openstreetmap.josm.data.coor.LatLon;
import org.openstreetmap.josm.gui.ExtendedDialog;
import org.openstreetmap.josm.gui.widgets.HtmlPanel;
import org.openstreetmap.josm.tools.GBC;
import org.openstreetmap.josm.tools.WindowGeometry;
public class LatLonDialog extends ExtendedDialog {
private static final Color BG_COLOR_ERROR = new Color(255, 224, 224);
public JTabbedPane tabs;
private JTextArea taLatLon;
private JScrollPane spScroll;
private JRadioButton rbNodes;
private JRadioButton rbWay;
private JRadioButton rbClosedWay;
private ButtonGroup bgType;
private LatLon[] latLonCoordinates;
private static final double ZERO = 0.0;
private static final String DEG = "\u00B0";
private static final String MIN = "\u2032";
private static final String SEC = "\u2033";
private static final char N_TR = LatLon.NORTH.charAt(0);
private static final char S_TR = LatLon.SOUTH.charAt(0);
private static final char E_TR = LatLon.EAST.charAt(0);
private static final char W_TR = LatLon.WEST.charAt(0);
private static final Pattern p = Pattern.compile(
"([+|-]?\\d+[.,]\\d+)|" // (1)
+ "([+|-]?\\d+)|" // (2)
+ "("+DEG+"|o|deg)|" // (3)
+ "('|"+MIN+"|min)|" // (4)
+ "(\"|"+SEC+"|sec)|" // (5)
+ "(,|;)|" // (6)
+ "([NSEW"+N_TR+S_TR+E_TR+W_TR+"])|"// (7)
+ "\\s+|"
+ "(.+)");
protected JPanel buildLatLon() {
JPanel pnl = new JPanel(new GridBagLayout());
pnl.setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));
pnl.add(new JLabel(tr("Coordinates:")), GBC.std().insets(0, 10, 5, 0));
taLatLon = new JTextArea(5, 24);
taLatLon.getDocument().addDocumentListener(new CoordinateListener());
spScroll = new JScrollPane(taLatLon);
pnl.add(spScroll, GBC.eol().insets(0, 10, 0, 0).fill().weight(2.0, 2.0));
//Radio button setup
bgType = new ButtonGroup();
rbNodes = new JRadioButton("Nodes", true);
rbNodes.setActionCommand("nodes");
bgType.add(rbNodes);
pnl.add(rbNodes);
rbWay = new JRadioButton("Way");
rbWay.setActionCommand("way");
bgType.add(rbWay);
pnl.add(rbWay);
rbClosedWay = new JRadioButton("Closed Way (Area)");
rbClosedWay.setActionCommand("area");
bgType.add(rbClosedWay);
pnl.add(rbClosedWay, GBC.eol());
pnl.add(new JSeparator(), GBC.eol().fill(GBC.HORIZONTAL).insets(0, 5, 0, 5));
pnl.add(new HtmlPanel(
tr("Enter the coordinates for the new nodes, one for each line.<br/>"+
"If you enter two lines with the same coordinates there will be generated duplicate nodes.<br/>"+
"You can separate longitude and latitude with space, comma or semicolon.<br/>" +
"Use positive numbers or N, E characters to indicate North or East cardinal direction.<br/>" +
"For South and West cardinal directions you can use either negative numbers or S, W characters.<br/>" +
"Coordinate value can be in one of three formats:<ul>" +
"<li><i>degrees</i><tt>°</tt></li>" +
"<li><i>degrees</i><tt>°</tt> <i>minutes</i><tt>'</tt></li>" +
"<li><i>degrees</i><tt>°</tt> <i>minutes</i><tt>'</tt> <i>seconds</i><tt>"</tt></li>" +
"</ul>" +
"Symbols <tt>°</tt>, <tt>'</tt>, <tt>′</tt>, <tt>"</tt>, <tt>″</tt> are optional.<br/><br/>" +
"Some examples:<ul>" +
"<li>49.29918° 19.24788°</li>" +
"<li>N 49.29918 E 19.24788</li>" +
"<li>W 49°29.918' S 19°24.788'</li>" +
"<li>N 49°29'04" E 19°24'43"</li>" +
"<li>49.29918 N, 19.24788 E</li>" +
"<li>49°29'21" N 19°24'38" E</li>" +
"<li>49 29 51, 19 24 18</li>" +
"<li>49 29, 19 24</li>" +
"<li>E 49 29, N 19 24</li>" +
"<li>49° 29; 19° 24</li>" +
"<li>N 49° 29, W 19° 24</li>" +
"<li>49° 29.5 S, 19° 24.6 E</li>" +
"<li>N 49 29.918 E 19 15.88</li>" +
"<li>49 29.4 19 24.5</li>" +
"<li>-49 29.4 N -19 24.5 W</li></ul>" +
"<li>48 deg 42' 52.13\" N, 21 deg 11' 47.60\" E</li></ul>"
)),
GBC.eol().fill().weight(1.0, 1.0));
// parse and verify input on the fly
//
LatLonInputVerifier inputVerifier = new LatLonInputVerifier();
taLatLon.getDocument().addDocumentListener(inputVerifier);
// select the text in the field on focus
//
TextFieldFocusHandler focusHandler = new TextFieldFocusHandler();
taLatLon.addFocusListener(focusHandler);
return pnl;
}
protected void build() {
tabs = new JTabbedPane();
tabs.addTab(tr("Lat/Lon"), buildLatLon());
tabs.getModel().addChangeListener(new ChangeListener() {
@Override
public void stateChanged(ChangeEvent e) {
switch (tabs.getModel().getSelectedIndex()) {
case 0: parseLatLonUserInput(); break;
default: throw new AssertionError();
}
}
});
setContent(tabs, false);
}
public LatLonDialog(Component parent, String title, String help) {
super(Main.parent, tr("Add Node..."), new String[] {tr("Ok"), tr("Cancel")});
setButtonIcons(new String[] {"ok", "cancel"});
configureContextsensitiveHelp("/Action/AddNode", true);
build();
setCoordinates(null);
}
public void setCoordinates(LatLon[] ll) {
if (ll == null) {
ll = new LatLon[] {};
}
this.latLonCoordinates = ll;
String text = "";
for (LatLon latlon : ll) {
text = text + latlon.latToString(CoordinateFormat.getDefaultFormat())
+ " " + latlon.lonToString(CoordinateFormat.getDefaultFormat()) + "\n";
}
taLatLon.setText(text);
setOkEnabled(true);
}
public LatLon[] getCoordinates() {
return latLonCoordinates;
}
public LatLon[] getLatLonCoordinates() {
return latLonCoordinates;
}
public String getGeomType() {
return bgType.getSelection().getActionCommand();
}
protected void setErrorFeedback(JTextArea tf, String message) {
tf.setBorder(BorderFactory.createLineBorder(Color.RED, 1));
tf.setToolTipText(message);
tf.setBackground(BG_COLOR_ERROR);
}
protected void clearErrorFeedback(JTextArea tf, String message) {
tf.setBorder(UIManager.getBorder("TextField.border"));
tf.setToolTipText(message);
tf.setBackground(UIManager.getColor("TextField.background"));
}
protected Double parseDoubleFromUserInput(String input) {
if (input == null) return null;
// remove white space and an optional degree symbol
//
input = input.trim();
input = input.replaceAll(DEG, "");
// try to parse using the current locale
//
NumberFormat f = NumberFormat.getNumberInstance();
Number n = null;
ParsePosition pp = new ParsePosition(0);
n = f.parse(input, pp);
if (pp.getErrorIndex() >= 0 || pp.getIndex() < input.length()) {
// fall back - try to parse with the english locale
//
pp = new ParsePosition(0);
f = NumberFormat.getNumberInstance(Locale.ENGLISH);
n = f.parse(input, pp);
if (pp.getErrorIndex() >= 0 || pp.getIndex() < input.length())
return null;
}
return n == null ? null : n.doubleValue();
}
protected void parseLatLonUserInput() {
LatLon[] latLons;
try {
latLons = parseLatLons(taLatLon.getText());
Boolean working = true;
int i = 0;
while (working && i < latLons.length) {
if (!LatLon.isValidLat(latLons[i].lat()) || !LatLon.isValidLon(latLons[i].lon())) {
latLons = null;
working = false;
}
i++;
}
} catch (IllegalArgumentException e) {
latLons = null;
}
if (latLons == null) {
setErrorFeedback(taLatLon, tr("Please enter a GPS coordinates"));
latLonCoordinates = null;
setOkEnabled(false);
} else {
clearErrorFeedback(taLatLon, tr("Please enter a GPS coordinates"));
latLonCoordinates = latLons;
setOkEnabled(true);
}
}
private void setOkEnabled(boolean b) {
if (buttons != null && buttons.size() > 0) {
buttons.get(0).setEnabled(b);
}
}
@Override
public void setVisible(boolean visible) {
final String preferenceKey = getClass().getName() + ".geometry";
if (visible) {
new WindowGeometry(
preferenceKey,
WindowGeometry.centerInWindow(getParent(), getSize())
).applySafe(this);
} else {
new WindowGeometry(this).remember(preferenceKey);
}
super.setVisible(visible);
}
class LatLonInputVerifier implements DocumentListener {
@Override public void changedUpdate(DocumentEvent e) {
parseLatLonUserInput();
}
@Override public void insertUpdate(DocumentEvent e) {
parseLatLonUserInput();
}
@Override public void removeUpdate(DocumentEvent e) {
parseLatLonUserInput();
}
}
static class TextFieldFocusHandler implements FocusListener {
@Override
public void focusGained(FocusEvent e) {
Component c = e.getComponent();
if (c instanceof JTextArea) {
JTextArea tf = (JTextArea) c;
tf.selectAll();
}
}
@Override
public void focusLost(FocusEvent e) {}
}
private static LatLon[] parseLatLons(final String text) {
String[] lines = text.split("\\r?\\n");
List<LatLon> latLons = new ArrayList<>();
for (String line : lines) {
latLons.add(parseLatLon(line));
}
return latLons.toArray(new LatLon[]{});
}
private static LatLon parseLatLon(final String coord) {
final Matcher m = p.matcher(coord);
final StringBuilder sb = new StringBuilder();
final List<Object> list = new ArrayList<>();
while (m.find()) {
if (m.group(1) != null) {
sb.append('R'); // floating point number
list.add(Double.parseDouble(m.group(1).replace(',', '.')));
} else if (m.group(2) != null) {
sb.append('Z'); // integer number
list.add(Double.parseDouble(m.group(2)));
} else if (m.group(3) != null) {
sb.append('o'); // degree sign
} else if (m.group(4) != null) {
sb.append('\''); // seconds sign
} else if (m.group(5) != null) {
sb.append('"'); // minutes sign
} else if (m.group(6) != null) {
sb.append(','); // separator
} else if (m.group(7) != null) {
sb.append("x"); // cardinal direction
String c = m.group(7).toUpperCase();
if (c.equals("N") || c.equals("S") || c.equals("E") || c.equals("W")) {
list.add(c);
} else {
list.add(c.replace(N_TR, 'N').replace(S_TR, 'S')
.replace(E_TR, 'E').replace(W_TR, 'W'));
}
} else if (m.group(8) != null) {
throw new IllegalArgumentException("invalid token: " + m.group(8));
}
}
final String pattern = sb.toString();
final Object[] params = list.toArray();
final LatLonHolder latLon = new LatLonHolder();
if (pattern.matches("Ro?,?Ro?")) {
setLatLonObj(latLon,
params[0], ZERO, ZERO, "N",
params[1], ZERO, ZERO, "E");
} else if (pattern.matches("xRo?,?xRo?")) {
setLatLonObj(latLon,
params[1], ZERO, ZERO, params[0],
params[3], ZERO, ZERO, params[2]);
} else if (pattern.matches("Ro?x,?Ro?x")) {
setLatLonObj(latLon,
params[0], ZERO, ZERO, params[1],
params[2], ZERO, ZERO, params[3]);
} else if (pattern.matches("Zo[RZ]'?,?Zo[RZ]'?|Z[RZ],?Z[RZ]")) {
setLatLonObj(latLon,
params[0], params[1], ZERO, "N",
params[2], params[3], ZERO, "E");
} else if (pattern.matches("xZo[RZ]'?,?xZo[RZ]'?|xZo?[RZ],?xZo?[RZ]")) {
setLatLonObj(latLon,
params[1], params[2], ZERO, params[0],
params[4], params[5], ZERO, params[3]);
} else if (pattern.matches("Zo[RZ]'?x,?Zo[RZ]'?x|Zo?[RZ]x,?Zo?[RZ]x")) {
setLatLonObj(latLon,
params[0], params[1], ZERO, params[2],
params[3], params[4], ZERO, params[5]);
} else if (pattern.matches("ZoZ'[RZ]\"?x,?ZoZ'[RZ]\"?x|ZZ[RZ]x,?ZZ[RZ]x")) {
setLatLonObj(latLon,
params[0], params[1], params[2], params[3],
params[4], params[5], params[6], params[7]);
} else if (pattern.matches("xZoZ'[RZ]\"?,?xZoZ'[RZ]\"?|xZZ[RZ],?xZZ[RZ]")) {
setLatLonObj(latLon,
params[1], params[2], params[3], params[0],
params[5], params[6], params[7], params[4]);
} else if (pattern.matches("ZZ[RZ],?ZZ[RZ]")) {
setLatLonObj(latLon,
params[0], params[1], params[2], "N",
params[3], params[4], params[5], "E");
} else {
throw new IllegalArgumentException("invalid format: " + pattern);
}
return new LatLon(latLon.lat, latLon.lon);
}
private static class LatLonHolder {
double lat, lon;
}
private static void setLatLonObj(final LatLonHolder latLon,
final Object coord1deg, final Object coord1min, final Object coord1sec, final Object card1,
final Object coord2deg, final Object coord2min, final Object coord2sec, final Object card2) {
setLatLon(latLon,
(Double) coord1deg, (Double) coord1min, (Double) coord1sec, (String) card1,
(Double) coord2deg, (Double) coord2min, (Double) coord2sec, (String) card2);
}
private static void setLatLon(final LatLonHolder latLon,
final double coord1deg, final double coord1min, final double coord1sec, final String card1,
final double coord2deg, final double coord2min, final double coord2sec, final String card2) {
setLatLon(latLon, coord1deg, coord1min, coord1sec, card1);
setLatLon(latLon, coord2deg, coord2min, coord2sec, card2);
}
private static void setLatLon(final LatLonHolder latLon, final double coordDeg, final double coordMin, final double coordSec,
final String card) {
if (coordDeg < -180 || coordDeg > 180 || coordMin < 0 || coordMin >= 60 || coordSec < 0 || coordSec > 60) {
throw new IllegalArgumentException("out of range");
}
double coord = (coordDeg < 0 ? -1 : 1) * (Math.abs(coordDeg) + coordMin / 60 + coordSec / 3600);
coord = card.equals("N") || card.equals("E") ? coord : -coord;
if (card.equals("N") || card.equals("S")) {
latLon.lat = coord;
} else {
latLon.lon = coord;
}
}
public String getLatLonText() {
return taLatLon.getText();
}
public void setLatLonText(String text) {
taLatLon.setText(text);
}
private class CoordinateListener implements DocumentListener {
@Override public void changedUpdate(DocumentEvent e) {
//not fired
}
@Override public void insertUpdate(DocumentEvent e) {
updateButtons();
}
@Override public void removeUpdate(DocumentEvent e) {
updateButtons();
}
private void updateButtons() {
String text = taLatLon.getText();
String[] lines = text.split("\r\n|\r|\n");
rbNodes.setEnabled(true);
rbWay.setEnabled(true);
rbClosedWay.setEnabled(true);
if (lines.length < 3) {
rbClosedWay.setEnabled(false);
bgType.setSelected(rbNodes.getModel(), true);
}
if (lines.length < 2) {
rbWay.setEnabled(false);
bgType.setSelected(rbNodes.getModel(), true);
}
}
}
}