source: josm/trunk/src/org/openstreetmap/josm/actions/PasteAction.java@ 10420

Last change on this file since 10420 was 10413, checked in by Don-vip, 8 years ago

fix #12983 - replace calls to Main.main.get[Active|Edit]Layer() by Main.getLayerManager().get[Active|Edit]Layer() - gsoc-core

  • Property svn:eol-style set to native
File size: 9.0 KB
Line 
1// License: GPL. For details, see LICENSE file.
2// Author: David Earl
3package org.openstreetmap.josm.actions;
4
5import static org.openstreetmap.josm.gui.help.HelpUtil.ht;
6import static org.openstreetmap.josm.tools.I18n.tr;
7
8import java.awt.MouseInfo;
9import java.awt.Point;
10import java.awt.event.ActionEvent;
11import java.awt.event.KeyEvent;
12import java.util.ArrayList;
13import java.util.HashMap;
14import java.util.List;
15import java.util.Map;
16
17import org.openstreetmap.josm.Main;
18import org.openstreetmap.josm.command.AddPrimitivesCommand;
19import org.openstreetmap.josm.data.coor.EastNorth;
20import org.openstreetmap.josm.data.osm.NodeData;
21import org.openstreetmap.josm.data.osm.OsmPrimitiveType;
22import org.openstreetmap.josm.data.osm.PrimitiveData;
23import org.openstreetmap.josm.data.osm.PrimitiveDeepCopy;
24import org.openstreetmap.josm.data.osm.PrimitiveDeepCopy.PasteBufferChangedListener;
25import org.openstreetmap.josm.data.osm.RelationData;
26import org.openstreetmap.josm.data.osm.RelationMemberData;
27import org.openstreetmap.josm.data.osm.WayData;
28import org.openstreetmap.josm.gui.ExtendedDialog;
29import org.openstreetmap.josm.gui.layer.Layer;
30import org.openstreetmap.josm.tools.Shortcut;
31
32/**
33 * Paste OSM primitives from clipboard to the current edit layer.
34 * @since 404
35 */
36public final class PasteAction extends JosmAction implements PasteBufferChangedListener {
37
38 /**
39 * Constructs a new {@code PasteAction}.
40 */
41 public PasteAction() {
42 super(tr("Paste"), "paste", tr("Paste contents of paste buffer."),
43 Shortcut.registerShortcut("system:paste", tr("Edit: {0}", tr("Paste")), KeyEvent.VK_V, Shortcut.CTRL), true);
44 putValue("help", ht("/Action/Paste"));
45 // CUA shortcut for paste (https://en.wikipedia.org/wiki/IBM_Common_User_Access#Description)
46 Main.registerActionShortcut(this,
47 Shortcut.registerShortcut("system:paste:cua", tr("Edit: {0}", tr("Paste")), KeyEvent.VK_INSERT, Shortcut.SHIFT));
48 Main.pasteBuffer.addPasteBufferChangedListener(this);
49 }
50
51 @Override
52 public void actionPerformed(ActionEvent e) {
53 if (!isEnabled())
54 return;
55 pasteData(Main.pasteBuffer, Main.pasteSource, e);
56 }
57
58 /**
59 * Paste OSM primitives from the given paste buffer and OSM data layer source to the current edit layer.
60 * @param pasteBuffer The paste buffer containing primitive ids to copy
61 * @param source The OSM data layer used to look for primitive ids
62 * @param e The ActionEvent that triggered this operation
63 */
64 public void pasteData(PrimitiveDeepCopy pasteBuffer, Layer source, ActionEvent e) {
65 /* Find the middle of the pasteBuffer area */
66 double maxEast = -1E100;
67 double minEast = 1E100;
68 double maxNorth = -1E100;
69 double minNorth = 1E100;
70 boolean incomplete = false;
71 for (PrimitiveData data : pasteBuffer.getAll()) {
72 if (data instanceof NodeData) {
73 NodeData n = (NodeData) data;
74 if (n.getEastNorth() != null) {
75 double east = n.getEastNorth().east();
76 double north = n.getEastNorth().north();
77 if (east > maxEast) {
78 maxEast = east;
79 }
80 if (east < minEast) {
81 minEast = east;
82 }
83 if (north > maxNorth) {
84 maxNorth = north;
85 }
86 if (north < minNorth) {
87 minNorth = north;
88 }
89 }
90 }
91 if (data.isIncomplete()) {
92 incomplete = true;
93 }
94 }
95
96 // Allow to cancel paste if there are incomplete primitives
97 if (incomplete && !confirmDeleteIncomplete()) {
98 return;
99 }
100
101 // default to paste in center of map (pasted via menu or cursor not in MapView)
102 EastNorth mPosition = Main.map.mapView.getCenter();
103 // We previously checked for modifier to know if the action has been trigerred via shortcut or via menu
104 // But this does not work if the shortcut is changed to a single key (see #9055)
105 // Observed behaviour: getActionCommand() returns Action.NAME when triggered via menu, but shortcut text when triggered with it
106 if (e != null && !getValue(NAME).equals(e.getActionCommand())) {
107 final Point mp = MouseInfo.getPointerInfo().getLocation();
108 final Point tl = Main.map.mapView.getLocationOnScreen();
109 final Point pos = new Point(mp.x-tl.x, mp.y-tl.y);
110 if (Main.map.mapView.contains(pos)) {
111 mPosition = Main.map.mapView.getEastNorth(pos.x, pos.y);
112 }
113 }
114
115 double offsetEast = mPosition.east() - (maxEast + minEast)/2.0;
116 double offsetNorth = mPosition.north() - (maxNorth + minNorth)/2.0;
117
118 // Make a copy of pasteBuffer and map from old id to copied data id
119 List<PrimitiveData> bufferCopy = new ArrayList<>();
120 List<PrimitiveData> toSelect = new ArrayList<>();
121 Map<Long, Long> newNodeIds = new HashMap<>();
122 Map<Long, Long> newWayIds = new HashMap<>();
123 Map<Long, Long> newRelationIds = new HashMap<>();
124 for (PrimitiveData data: pasteBuffer.getAll()) {
125 if (data.isIncomplete()) {
126 continue;
127 }
128 PrimitiveData copy = data.makeCopy();
129 copy.clearOsmMetadata();
130 if (data instanceof NodeData) {
131 newNodeIds.put(data.getUniqueId(), copy.getUniqueId());
132 } else if (data instanceof WayData) {
133 newWayIds.put(data.getUniqueId(), copy.getUniqueId());
134 } else if (data instanceof RelationData) {
135 newRelationIds.put(data.getUniqueId(), copy.getUniqueId());
136 }
137 bufferCopy.add(copy);
138 if (pasteBuffer.getDirectlyAdded().contains(data)) {
139 toSelect.add(copy);
140 }
141 }
142
143 // Update references in copied buffer
144 for (PrimitiveData data:bufferCopy) {
145 if (data instanceof NodeData) {
146 NodeData nodeData = (NodeData) data;
147 if (Main.getLayerManager().getEditLayer() == source) {
148 nodeData.setEastNorth(nodeData.getEastNorth().add(offsetEast, offsetNorth));
149 }
150 } else if (data instanceof WayData) {
151 List<Long> newNodes = new ArrayList<>();
152 for (Long oldNodeId: ((WayData) data).getNodes()) {
153 Long newNodeId = newNodeIds.get(oldNodeId);
154 if (newNodeId != null) {
155 newNodes.add(newNodeId);
156 }
157 }
158 ((WayData) data).setNodes(newNodes);
159 } else if (data instanceof RelationData) {
160 List<RelationMemberData> newMembers = new ArrayList<>();
161 for (RelationMemberData member: ((RelationData) data).getMembers()) {
162 OsmPrimitiveType memberType = member.getMemberType();
163 Long newId;
164 switch (memberType) {
165 case NODE:
166 newId = newNodeIds.get(member.getMemberId());
167 break;
168 case WAY:
169 newId = newWayIds.get(member.getMemberId());
170 break;
171 case RELATION:
172 newId = newRelationIds.get(member.getMemberId());
173 break;
174 default: throw new AssertionError();
175 }
176 if (newId != null) {
177 newMembers.add(new RelationMemberData(member.getRole(), memberType, newId));
178 }
179 }
180 ((RelationData) data).setMembers(newMembers);
181 }
182 }
183
184 /* Now execute the commands to add the duplicated contents of the paste buffer to the map */
185 Main.main.undoRedo.add(new AddPrimitivesCommand(bufferCopy, toSelect));
186 Main.map.mapView.repaint();
187 }
188
189 private static boolean confirmDeleteIncomplete() {
190 ExtendedDialog ed = new ExtendedDialog(Main.parent,
191 tr("Delete incomplete members?"),
192 new String[] {tr("Paste without incomplete members"), tr("Cancel")});
193 ed.setButtonIcons(new String[] {"dialogs/relation/deletemembers", "cancel"});
194 ed.setContent(tr("The copied data contains incomplete objects. "
195 + "When pasting the incomplete objects are removed. "
196 + "Do you want to paste the data without the incomplete objects?"));
197 ed.showDialog();
198 return ed.getValue() == 1;
199 }
200
201 @Override
202 protected void updateEnabledState() {
203 if (getLayerManager().getEditDataSet() == null || Main.pasteBuffer == null) {
204 setEnabled(false);
205 return;
206 }
207 setEnabled(!Main.pasteBuffer.isEmpty());
208 }
209
210 @Override
211 public void pasteBufferChanged(PrimitiveDeepCopy pasteBuffer) {
212 updateEnabledState();
213 }
214}
Note: See TracBrowser for help on using the repository browser.