// License: GPL. For details, see LICENSE file. package org.openstreetmap.josm.actions; import static org.openstreetmap.josm.gui.help.HelpUtil.ht; import static org.openstreetmap.josm.tools.I18n.marktr; import static org.openstreetmap.josm.tools.I18n.tr; import java.awt.event.ActionEvent; import java.awt.event.KeyEvent; import java.awt.geom.Area; import java.util.ArrayList; import java.util.Arrays; import java.util.Collection; import java.util.Collections; import java.util.HashSet; import java.util.List; import java.util.concurrent.TimeUnit; import javax.swing.JOptionPane; import javax.swing.event.ListSelectionListener; import javax.swing.event.TreeSelectionListener; import org.openstreetmap.josm.Main; import org.openstreetmap.josm.data.Bounds; import org.openstreetmap.josm.data.DataSource; import org.openstreetmap.josm.data.conflict.Conflict; import org.openstreetmap.josm.data.osm.DataSet; import org.openstreetmap.josm.data.osm.OsmPrimitive; import org.openstreetmap.josm.data.osm.visitor.BoundingXYVisitor; import org.openstreetmap.josm.data.validation.TestError; import org.openstreetmap.josm.gui.MainApplication; import org.openstreetmap.josm.gui.MapFrame; import org.openstreetmap.josm.gui.MapFrameListener; import org.openstreetmap.josm.gui.MapView; import org.openstreetmap.josm.gui.dialogs.ConflictDialog; import org.openstreetmap.josm.gui.dialogs.LayerListDialog; import org.openstreetmap.josm.gui.dialogs.ValidatorDialog.ValidatorBoundingXYVisitor; import org.openstreetmap.josm.gui.layer.Layer; import org.openstreetmap.josm.spi.preferences.Config; import org.openstreetmap.josm.tools.Logging; import org.openstreetmap.josm.tools.Shortcut; /** * Toggles the autoScale feature of the mapView * @author imi */ public class AutoScaleAction extends JosmAction { /** * A list of things we can zoom to. The zoom target is given depending on the mode. */ public static final Collection MODES = Collections.unmodifiableList(Arrays.asList( marktr(/* ICON(dialogs/autoscale/) */ "data"), marktr(/* ICON(dialogs/autoscale/) */ "layer"), marktr(/* ICON(dialogs/autoscale/) */ "selection"), marktr(/* ICON(dialogs/autoscale/) */ "conflict"), marktr(/* ICON(dialogs/autoscale/) */ "download"), marktr(/* ICON(dialogs/autoscale/) */ "problem"), marktr(/* ICON(dialogs/autoscale/) */ "previous"), marktr(/* ICON(dialogs/autoscale/) */ "next"))); /** * One of {@link #MODES}. Defines what we are zooming to. */ private final String mode; /** Time of last zoom to bounds action */ protected long lastZoomTime = -1; /** Last zommed bounds */ protected int lastZoomArea = -1; /** * Zooms the current map view to the currently selected primitives. * Does nothing if there either isn't a current map view or if there isn't a current data layer. * */ public static void zoomToSelection() { DataSet dataSet = MainApplication.getLayerManager().getActiveDataSet(); if (dataSet == null) { return; } Collection sel = dataSet.getSelected(); if (sel.isEmpty()) { JOptionPane.showMessageDialog( Main.parent, tr("Nothing selected to zoom to."), tr("Information"), JOptionPane.INFORMATION_MESSAGE); return; } zoomTo(sel); } /** * Zooms the view to display the given set of primitives. * @param sel The primitives to zoom to, e.g. the current selection. */ public static void zoomTo(Collection sel) { BoundingXYVisitor bboxCalculator = new BoundingXYVisitor(); bboxCalculator.computeBoundingBox(sel); // increase bbox. This is required // especially if the bbox contains one single node, but helpful // in most other cases as well. bboxCalculator.enlargeBoundingBox(); if (bboxCalculator.getBounds() != null) { MainApplication.getMap().mapView.zoomTo(bboxCalculator); } } /** * Performs the auto scale operation of the given mode without the need to create a new action. * @param mode One of {@link #MODES}. */ public static void autoScale(String mode) { new AutoScaleAction(mode, false).autoScale(); } private static int getModeShortcut(String mode) { int shortcut = -1; // TODO: convert this to switch/case and make sure the parsing still works // CHECKSTYLE.OFF: LeftCurly // CHECKSTYLE.OFF: RightCurly /* leave as single line for shortcut overview parsing! */ if (mode.equals("data")) { shortcut = KeyEvent.VK_1; } else if (mode.equals("layer")) { shortcut = KeyEvent.VK_2; } else if (mode.equals("selection")) { shortcut = KeyEvent.VK_3; } else if (mode.equals("conflict")) { shortcut = KeyEvent.VK_4; } else if (mode.equals("download")) { shortcut = KeyEvent.VK_5; } else if (mode.equals("problem")) { shortcut = KeyEvent.VK_6; } else if (mode.equals("previous")) { shortcut = KeyEvent.VK_8; } else if (mode.equals("next")) { shortcut = KeyEvent.VK_9; } // CHECKSTYLE.ON: LeftCurly // CHECKSTYLE.ON: RightCurly return shortcut; } /** * Constructs a new {@code AutoScaleAction}. * @param mode The autoscale mode (one of {@link AutoScaleAction#MODES}) * @param marker Must be set to false. Used only to differentiate from default constructor */ private AutoScaleAction(String mode, boolean marker) { super(marker); this.mode = mode; } /** * Constructs a new {@code AutoScaleAction}. * @param mode The autoscale mode (one of {@link AutoScaleAction#MODES}) */ public AutoScaleAction(final String mode) { super(tr("Zoom to {0}", tr(mode)), "dialogs/autoscale/" + mode, tr("Zoom the view to {0}.", tr(mode)), Shortcut.registerShortcut("view:zoom" + mode, tr("View: {0}", tr("Zoom to {0}", tr(mode))), getModeShortcut(mode), Shortcut.DIRECT), true, null, false); String modeHelp = Character.toUpperCase(mode.charAt(0)) + mode.substring(1); putValue("help", "Action/AutoScale/" + modeHelp); this.mode = mode; switch (mode) { case "data": putValue("help", ht("/Action/ZoomToData")); break; case "layer": putValue("help", ht("/Action/ZoomToLayer")); break; case "selection": putValue("help", ht("/Action/ZoomToSelection")); break; case "conflict": putValue("help", ht("/Action/ZoomToConflict")); break; case "problem": putValue("help", ht("/Action/ZoomToProblem")); break; case "download": putValue("help", ht("/Action/ZoomToDownload")); break; case "previous": putValue("help", ht("/Action/ZoomToPrevious")); break; case "next": putValue("help", ht("/Action/ZoomToNext")); break; default: throw new IllegalArgumentException("Unknown mode: " + mode); } installAdapters(); } /** * Performs this auto scale operation for the mode this action is in. */ public void autoScale() { if (MainApplication.isDisplayingMapView()) { MapView mapView = MainApplication.getMap().mapView; switch (mode) { case "previous": mapView.zoomPrevious(); break; case "next": mapView.zoomNext(); break; default: BoundingXYVisitor bbox = getBoundingBox(); if (bbox != null && bbox.getBounds() != null) { mapView.zoomTo(bbox); } } } putValue("active", Boolean.TRUE); } @Override public void actionPerformed(ActionEvent e) { autoScale(); } /** * Replies the first selected layer in the layer list dialog. null, if no * such layer exists, either because the layer list dialog is not yet created * or because no layer is selected. * * @return the first selected layer in the layer list dialog */ protected Layer getFirstSelectedLayer() { if (getLayerManager().getActiveLayer() == null) { return null; } try { List layers = LayerListDialog.getInstance().getModel().getSelectedLayers(); if (!layers.isEmpty()) return layers.get(0); } catch (IllegalStateException e) { Logging.error(e); } return null; } private BoundingXYVisitor getBoundingBox() { switch (mode) { case "problem": return modeProblem(new ValidatorBoundingXYVisitor()); case "data": return modeData(new BoundingXYVisitor()); case "layer": return modeLayer(new BoundingXYVisitor()); case "selection": case "conflict": return modeSelectionOrConflict(new BoundingXYVisitor()); case "download": return modeDownload(new BoundingXYVisitor()); default: return new BoundingXYVisitor(); } } private static BoundingXYVisitor modeProblem(ValidatorBoundingXYVisitor v) { TestError error = MainApplication.getMap().validatorDialog.getSelectedError(); if (error == null) return null; v.visit(error); if (v.getBounds() == null) return null; v.enlargeBoundingBox(Config.getPref().getDouble("validator.zoom-enlarge-bbox", 0.0002)); return v; } private static BoundingXYVisitor modeData(BoundingXYVisitor v) { for (Layer l : MainApplication.getLayerManager().getLayers()) { l.visitBoundingBox(v); } return v; } private BoundingXYVisitor modeLayer(BoundingXYVisitor v) { // try to zoom to the first selected layer Layer l = getFirstSelectedLayer(); if (l == null) return null; l.visitBoundingBox(v); return v; } private BoundingXYVisitor modeSelectionOrConflict(BoundingXYVisitor v) { Collection sel = new HashSet<>(); if ("selection".equals(mode)) { DataSet dataSet = getLayerManager().getActiveDataSet(); if (dataSet != null) { sel = dataSet.getSelected(); } } else { ConflictDialog conflictDialog = MainApplication.getMap().conflictDialog; Conflict c = conflictDialog.getSelectedConflict(); if (c != null) { sel.add(c.getMy()); } else if (conflictDialog.getConflicts() != null) { sel = conflictDialog.getConflicts().getMyConflictParties(); } } if (sel.isEmpty()) { JOptionPane.showMessageDialog( Main.parent, "selection".equals(mode) ? tr("Nothing selected to zoom to.") : tr("No conflicts to zoom to"), tr("Information"), JOptionPane.INFORMATION_MESSAGE); return null; } for (OsmPrimitive osm : sel) { osm.accept(v); } // Increase the bounding box by up to 100% to give more context. v.enlargeBoundingBoxLogarithmically(100); // Make the bounding box at least 100 meter wide to // ensure reasonable zoom level when zooming onto single nodes. v.enlargeToMinSize(Config.getPref().getDouble("zoom_to_selection_min_size_in_meter", 100)); return v; } private BoundingXYVisitor modeDownload(BoundingXYVisitor v) { if (lastZoomTime > 0 && System.currentTimeMillis() - lastZoomTime > Config.getPref().getLong("zoom.bounds.reset.time", TimeUnit.SECONDS.toMillis(10))) { lastZoomTime = -1; } final DataSet dataset = getLayerManager().getActiveDataSet(); if (dataset != null) { List dataSources = new ArrayList<>(dataset.getDataSources()); int s = dataSources.size(); if (s > 0) { if (lastZoomTime == -1 || lastZoomArea == -1 || lastZoomArea > s) { lastZoomArea = s-1; v.visit(dataSources.get(lastZoomArea).bounds); } else if (lastZoomArea > 0) { lastZoomArea -= 1; v.visit(dataSources.get(lastZoomArea).bounds); } else { lastZoomArea = -1; Area sourceArea = getLayerManager().getActiveDataSet().getDataSourceArea(); if (sourceArea != null) { v.visit(new Bounds(sourceArea.getBounds2D())); } } lastZoomTime = System.currentTimeMillis(); } else { lastZoomTime = -1; lastZoomArea = -1; } } return v; } @Override protected void updateEnabledState() { DataSet ds = getLayerManager().getActiveDataSet(); MapFrame map = MainApplication.getMap(); switch (mode) { case "selection": setEnabled(ds != null && !ds.selectionEmpty()); break; case "layer": setEnabled(getFirstSelectedLayer() != null); break; case "conflict": setEnabled(map != null && map.conflictDialog.getSelectedConflict() != null); break; case "download": setEnabled(ds != null && !ds.getDataSources().isEmpty()); break; case "problem": setEnabled(map != null && map.validatorDialog.getSelectedError() != null); break; case "previous": setEnabled(MainApplication.isDisplayingMapView() && map.mapView.hasZoomUndoEntries()); break; case "next": setEnabled(MainApplication.isDisplayingMapView() && map.mapView.hasZoomRedoEntries()); break; default: setEnabled(!getLayerManager().getLayers().isEmpty()); } } @Override protected void updateEnabledState(Collection selection) { if ("selection".equals(mode)) { setEnabled(selection != null && !selection.isEmpty()); } } @Override protected final void installAdapters() { super.installAdapters(); // make this action listen to zoom and mapframe change events // MapView.addZoomChangeListener(new ZoomChangeAdapter()); MainApplication.addMapFrameListener(new MapFrameAdapter()); initEnabledState(); } /** * Adapter for zoom change events */ private class ZoomChangeAdapter implements MapView.ZoomChangeListener { @Override public void zoomChanged() { updateEnabledState(); } } /** * Adapter for MapFrame change events */ private class MapFrameAdapter implements MapFrameListener { private ListSelectionListener conflictSelectionListener; private TreeSelectionListener validatorSelectionListener; MapFrameAdapter() { if ("conflict".equals(mode)) { conflictSelectionListener = e -> updateEnabledState(); } else if ("problem".equals(mode)) { validatorSelectionListener = e -> updateEnabledState(); } } @Override public void mapFrameInitialized(MapFrame oldFrame, MapFrame newFrame) { if (conflictSelectionListener != null) { if (newFrame != null) { newFrame.conflictDialog.addListSelectionListener(conflictSelectionListener); } else if (oldFrame != null) { oldFrame.conflictDialog.removeListSelectionListener(conflictSelectionListener); } } else if (validatorSelectionListener != null) { if (newFrame != null) { newFrame.validatorDialog.addTreeSelectionListener(validatorSelectionListener); } else if (oldFrame != null) { oldFrame.validatorDialog.removeTreeSelectionListener(validatorSelectionListener); } } updateEnabledState(); } } }