source: josm/trunk/src/org/openstreetmap/josm/actions/MergeLayerAction.java@ 15450

Last change on this file since 15450 was 15450, checked in by Don-vip, 5 years ago

fix #18216 - Adapt 'Merge' button text to 'Merge layer' and 'Merge selection' actions

  • Property svn:eol-style set to native
File size: 8.5 KB
Line 
1// License: GPL. For details, see LICENSE file.
2package org.openstreetmap.josm.actions;
3
4import static org.openstreetmap.josm.gui.help.HelpUtil.ht;
5import static org.openstreetmap.josm.tools.I18n.tr;
6
7import java.awt.event.ActionEvent;
8import java.awt.event.KeyEvent;
9import java.util.ArrayList;
10import java.util.Collection;
11import java.util.Collections;
12import java.util.List;
13import java.util.concurrent.Future;
14
15import org.openstreetmap.josm.gui.MainApplication;
16import org.openstreetmap.josm.gui.dialogs.LayerListDialog;
17import org.openstreetmap.josm.gui.dialogs.layer.MergeGpxLayerDialog;
18import org.openstreetmap.josm.gui.layer.GpxLayer;
19import org.openstreetmap.josm.gui.layer.Layer;
20import org.openstreetmap.josm.gui.layer.OsmDataLayer;
21import org.openstreetmap.josm.gui.util.GuiHelper;
22import org.openstreetmap.josm.spi.preferences.Config;
23import org.openstreetmap.josm.tools.ImageProvider;
24import org.openstreetmap.josm.tools.Logging;
25import org.openstreetmap.josm.tools.Shortcut;
26import org.openstreetmap.josm.tools.Utils;
27
28/**
29 * Action that merges two or more OSM data layers.
30 * @since 1890
31 */
32public class MergeLayerAction extends AbstractMergeAction {
33
34 /**
35 * Constructs a new {@code MergeLayerAction}.
36 */
37 public MergeLayerAction() {
38 super(tr("Merge layer"), "dialogs/mergedown",
39 tr("Merge the current layer into another layer"),
40 Shortcut.registerShortcut("system:merge", tr("Edit: {0}",
41 tr("Merge")), KeyEvent.VK_M, Shortcut.CTRL),
42 true, "action/mergelayer", true);
43 setHelpId(ht("/Action/MergeLayer"));
44 }
45
46 /**
47 * Submits merge of layers.
48 * @param targetLayers possible target layers
49 * @param sourceLayers source layers
50 * @return a Future representing pending completion of the merge task, or {@code null}
51 * @since 11885 (return type)
52 */
53 protected Future<?> doMerge(List<Layer> targetLayers, final Collection<Layer> sourceLayers) {
54 final boolean onlygpx = targetLayers.stream().noneMatch(l -> !(l instanceof GpxLayer));
55 final TargetLayerDialogResult<Layer> res = askTargetLayer(targetLayers, onlygpx,
56 tr("Cut timewise overlapping parts of tracks"),
57 onlygpx && Config.getPref().getBoolean("mergelayer.gpx.cut", false), tr("Merge layer"));
58 final Layer targetLayer = res.selectedTargetLayer;
59 if (targetLayer == null)
60 return null;
61
62 if (onlygpx) {
63 Config.getPref().putBoolean("mergelayer.gpx.cut", res.checkboxTicked);
64 }
65
66 final Object actionName = getValue(NAME);
67 if (onlygpx && res.checkboxTicked) {
68 List<GpxLayer> layers = new ArrayList<>();
69 layers.add((GpxLayer) targetLayer);
70 for (Layer sl : sourceLayers) {
71 if (sl != null && !sl.equals(targetLayer)) {
72 layers.add((GpxLayer) sl);
73 }
74 }
75 final MergeGpxLayerDialog d = new MergeGpxLayerDialog(MainApplication.getMainFrame(), layers);
76
77 if (d.showDialog().getValue() == 1) {
78
79 final boolean connect = d.connectCuts();
80 final List<GpxLayer> sortedLayers = d.getSortedLayers();
81
82 return MainApplication.worker.submit(() -> {
83 final long start = System.currentTimeMillis();
84
85 for (int i = sortedLayers.size() - 2; i >= 0; i--) {
86 final GpxLayer lower = sortedLayers.get(i + 1);
87 sortedLayers.get(i).mergeFrom(lower, true, connect);
88 GuiHelper.runInEDTAndWait(() -> getLayerManager().removeLayer(lower));
89 }
90
91 Logging.info(tr("{0} completed in {1}", actionName, Utils.getDurationString(System.currentTimeMillis() - start)));
92 });
93 }
94 }
95
96 return MainApplication.worker.submit(() -> {
97 final long start = System.currentTimeMillis();
98 boolean layerMerged = false;
99 for (final Layer sourceLayer: sourceLayers) {
100 if (sourceLayer != null && !sourceLayer.equals(targetLayer)) {
101 if (sourceLayer instanceof OsmDataLayer && targetLayer instanceof OsmDataLayer
102 && ((OsmDataLayer) sourceLayer).isUploadDiscouraged() != ((OsmDataLayer) targetLayer).isUploadDiscouraged()
103 && Boolean.TRUE.equals(GuiHelper.runInEDTAndWaitAndReturn(() ->
104 warnMergingUploadDiscouragedLayers(sourceLayer, targetLayer)))) {
105 break;
106 }
107 targetLayer.mergeFrom(sourceLayer);
108 GuiHelper.runInEDTAndWait(() -> getLayerManager().removeLayer(sourceLayer));
109 layerMerged = true;
110 }
111 }
112
113 if (layerMerged) {
114 getLayerManager().setActiveLayer(targetLayer);
115 Logging.info(tr("{0} completed in {1}", actionName, Utils.getDurationString(System.currentTimeMillis() - start)));
116 }
117 });
118 }
119
120 /**
121 * Merges a list of layers together.
122 * @param sourceLayers The layers to merge
123 * @return a Future representing pending completion of the merge task, or {@code null}
124 * @since 11885 (return type)
125 */
126 public Future<?> merge(List<Layer> sourceLayers) {
127 return doMerge(sourceLayers, sourceLayers);
128 }
129
130 /**
131 * Merges the given source layer with another one, determined at runtime.
132 * @param sourceLayer The source layer to merge
133 * @return a Future representing pending completion of the merge task, or {@code null}
134 * @since 11885 (return type)
135 */
136 public Future<?> merge(Layer sourceLayer) {
137 if (sourceLayer == null)
138 return null;
139 List<Layer> targetLayers = LayerListDialog.getInstance().getModel().getPossibleMergeTargets(sourceLayer);
140 if (targetLayers.isEmpty()) {
141 warnNoTargetLayersForSourceLayer(sourceLayer);
142 return null;
143 }
144 return doMerge(targetLayers, Collections.singleton(sourceLayer));
145 }
146
147 @Override
148 public void actionPerformed(ActionEvent e) {
149 merge(getSourceLayer());
150 }
151
152 @Override
153 protected void updateEnabledState() {
154 GuiHelper.runInEDT(() -> {
155 final Layer sourceLayer = getSourceLayer();
156 if (sourceLayer == null) {
157 setEnabled(false);
158 } else {
159 try {
160 setEnabled(!LayerListDialog.getInstance().getModel().getPossibleMergeTargets(sourceLayer).isEmpty());
161 } catch (IllegalStateException e) {
162 // May occur when destroying last layer / exiting JOSM, see #14476
163 setEnabled(false);
164 Logging.error(e);
165 }
166 }
167 });
168 }
169
170 /**
171 * Returns the source layer.
172 * @return the source layer
173 */
174 protected Layer getSourceLayer() {
175 return getLayerManager().getActiveLayer();
176 }
177
178 /**
179 * Warns about a discouraged merge operation, ask for confirmation.
180 * @param sourceLayer The source layer
181 * @param targetLayer The target layer
182 * @return {@code true} if the user wants to cancel, {@code false} if they want to continue
183 */
184 public static final boolean warnMergingUploadDiscouragedLayers(Layer sourceLayer, Layer targetLayer) {
185 return GuiHelper.warnUser(tr("Merging layers with different upload policies"),
186 "<html>" +
187 tr("You are about to merge data between layers ''{0}'' and ''{1}''.<br /><br />"+
188 "These layers have different upload policies and should not been merged as it.<br />"+
189 "Merging them will result to enforce the stricter policy (upload discouraged) to ''{1}''.<br /><br />"+
190 "<b>This is not the recommended way of merging such data</b>.<br />"+
191 "You should instead check and merge each object, one by one, by using ''<i>Merge selection</i>''.<br /><br />"+
192 "Are you sure you want to continue?",
193 Utils.escapeReservedCharactersHTML(sourceLayer.getName()),
194 Utils.escapeReservedCharactersHTML(targetLayer.getName()),
195 Utils.escapeReservedCharactersHTML(targetLayer.getName()))+
196 "</html>",
197 ImageProvider.get("dialogs", "mergedown"), tr("Ignore this hint and merge anyway"));
198 }
199}
Note: See TracBrowser for help on using the repository browser.