source: josm/trunk/src/org/openstreetmap/josm/gui/dialogs/properties/PropertiesDialog.java@ 5821

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

see #7846 - Large code refactorization in management of popup menus to simplify interactions with plugins (needed at least for imagery-xml-bounds and tag2link plugins)

  • Property svn:eol-style set to native
File size: 51.5 KB
Line 
1// License: GPL. For details, see LICENSE file.
2package org.openstreetmap.josm.gui.dialogs.properties;
3
4import static org.openstreetmap.josm.tools.I18n.tr;
5
6import java.awt.Component;
7import java.awt.Container;
8import java.awt.Font;
9import java.awt.GridBagLayout;
10import java.awt.Point;
11import java.awt.event.ActionEvent;
12import java.awt.event.KeyEvent;
13import java.awt.event.MouseAdapter;
14import java.awt.event.MouseEvent;
15import java.net.HttpURLConnection;
16import java.net.URI;
17import java.net.URLEncoder;
18import java.util.ArrayList;
19import java.util.Arrays;
20import java.util.Collection;
21import java.util.Collections;
22import java.util.Comparator;
23import java.util.EnumSet;
24import java.util.HashMap;
25import java.util.LinkedList;
26import java.util.List;
27import java.util.Map;
28import java.util.Map.Entry;
29import java.util.Set;
30import java.util.TreeMap;
31import java.util.TreeSet;
32
33import javax.swing.AbstractAction;
34import javax.swing.JComponent;
35import javax.swing.JLabel;
36import javax.swing.JPanel;
37import javax.swing.JPopupMenu;
38import javax.swing.JScrollPane;
39import javax.swing.JTable;
40import javax.swing.KeyStroke;
41import javax.swing.ListSelectionModel;
42import javax.swing.event.ListSelectionEvent;
43import javax.swing.event.ListSelectionListener;
44import javax.swing.table.DefaultTableCellRenderer;
45import javax.swing.table.DefaultTableModel;
46import javax.swing.table.TableColumnModel;
47import javax.swing.table.TableModel;
48
49import org.openstreetmap.josm.Main;
50import org.openstreetmap.josm.actions.JosmAction;
51import org.openstreetmap.josm.actions.relation.DownloadSelectedIncompleteMembersAction;
52import org.openstreetmap.josm.actions.relation.SelectMembersAction;
53import org.openstreetmap.josm.actions.relation.SelectRelationAction;
54import org.openstreetmap.josm.actions.search.SearchAction.SearchMode;
55import org.openstreetmap.josm.actions.search.SearchAction.SearchSetting;
56import org.openstreetmap.josm.command.ChangeCommand;
57import org.openstreetmap.josm.command.ChangePropertyCommand;
58import org.openstreetmap.josm.command.Command;
59import org.openstreetmap.josm.data.SelectionChangedListener;
60import org.openstreetmap.josm.data.osm.IRelation;
61import org.openstreetmap.josm.data.osm.Node;
62import org.openstreetmap.josm.data.osm.OsmPrimitive;
63import org.openstreetmap.josm.data.osm.Relation;
64import org.openstreetmap.josm.data.osm.RelationMember;
65import org.openstreetmap.josm.data.osm.Tag;
66import org.openstreetmap.josm.data.osm.Way;
67import org.openstreetmap.josm.data.osm.event.AbstractDatasetChangedEvent;
68import org.openstreetmap.josm.data.osm.event.DataSetListenerAdapter;
69import org.openstreetmap.josm.data.osm.event.DatasetEventManager;
70import org.openstreetmap.josm.data.osm.event.DatasetEventManager.FireMode;
71import org.openstreetmap.josm.data.osm.event.SelectionEventManager;
72import org.openstreetmap.josm.gui.DefaultNameFormatter;
73import org.openstreetmap.josm.gui.ExtendedDialog;
74import org.openstreetmap.josm.gui.MapFrame;
75import org.openstreetmap.josm.gui.MapView;
76import org.openstreetmap.josm.gui.PopupMenuHandler;
77import org.openstreetmap.josm.gui.SideButton;
78import org.openstreetmap.josm.gui.dialogs.ToggleDialog;
79import org.openstreetmap.josm.gui.dialogs.properties.PresetListPanel.PresetHandler;
80import org.openstreetmap.josm.gui.dialogs.relation.RelationEditor;
81import org.openstreetmap.josm.gui.layer.OsmDataLayer;
82import org.openstreetmap.josm.gui.tagging.TaggingPreset;
83import org.openstreetmap.josm.gui.tagging.TaggingPreset.PresetType;
84import org.openstreetmap.josm.gui.widgets.PopupMenuLauncher;
85import org.openstreetmap.josm.tools.GBC;
86import org.openstreetmap.josm.tools.ImageProvider;
87import org.openstreetmap.josm.tools.InputMapUtils;
88import org.openstreetmap.josm.tools.LanguageInfo;
89import org.openstreetmap.josm.tools.OpenBrowser;
90import org.openstreetmap.josm.tools.Shortcut;
91import org.openstreetmap.josm.tools.Utils;
92
93/**
94 * This dialog displays the properties of the current selected primitives.
95 *
96 * If no object is selected, the dialog list is empty.
97 * If only one is selected, all properties of this object are selected.
98 * If more than one object are selected, the sum of all properties are displayed. If the
99 * different objects share the same property, the shared value is displayed. If they have
100 * different values, all of them are put in a combo box and the string "<different>"
101 * is displayed in italic.
102 *
103 * Below the list, the user can click on an add, modify and delete property button to
104 * edit the table selection value.
105 *
106 * The command is applied to all selected entries.
107 *
108 * @author imi
109 */
110public class PropertiesDialog extends ToggleDialog implements SelectionChangedListener, MapView.EditLayerChangeListener, DataSetListenerAdapter.Listener {
111
112 /**
113 * hook for roadsigns plugin to display a small button in the upper right corner of this dialog
114 */
115 public static final JPanel pluginHook = new JPanel();
116
117 /**
118 * The property data of selected objects.
119 */
120 private final DefaultTableModel propertyData = new ReadOnlyTableModel();
121
122 /**
123 * The membership data of selected objects.
124 */
125 private final DefaultTableModel membershipData = new ReadOnlyTableModel();
126
127 /**
128 * The properties table.
129 */
130 private final JTable propertyTable = new JTable(propertyData);
131 /**
132 * The membership table.
133 */
134 private final JTable membershipTable = new JTable(membershipData);
135
136 // Popup menus
137 private final JPopupMenu propertyMenu = new JPopupMenu();
138 private final JPopupMenu membershipMenu = new JPopupMenu();
139
140 // Popup menu handlers
141 private final PopupMenuHandler propertyMenuHandler = new PopupMenuHandler(propertyMenu);
142 private final PopupMenuHandler membershipMenuHandler = new PopupMenuHandler(membershipMenu);
143
144 private final Map<String, Map<String, Integer>> valueCount = new TreeMap<String, Map<String, Integer>>();
145 /**
146 * This sub-object is responsible for all adding and editing of properties
147 */
148 private final TagEditHelper editHelper = new TagEditHelper(propertyData, valueCount);
149
150 private final DataSetListenerAdapter dataChangedAdapter = new DataSetListenerAdapter(this);
151 private final HelpAction helpAction = new HelpAction();
152 private final PasteValueAction pasteValueAction = new PasteValueAction();
153 private final CopyValueAction copyValueAction = new CopyValueAction();
154 private final CopyKeyValueAction copyKeyValueAction = new CopyKeyValueAction();
155 private final CopyAllKeyValueAction copyAllKeyValueAction = new CopyAllKeyValueAction();
156 private final SearchAction searchActionSame = new SearchAction(true);
157 private final SearchAction searchActionAny = new SearchAction(false);
158 private final AddAction addAction = new AddAction();
159 private final EditAction editAction = new EditAction();
160 private final DeleteAction deleteAction = new DeleteAction();
161 private final JosmAction[] josmActions = new JosmAction[]{addAction, editAction, deleteAction};
162
163 // relation actions
164 private final DownloadSelectedIncompleteMembersAction downloadSelectedIncompleteMembersAction = new DownloadSelectedIncompleteMembersAction();
165 private final SelectRelationAction addRelationToSelectionAction = new SelectRelationAction(true);
166 private final SelectMembersAction addMembersToSelectionAction = new SelectMembersAction(true);
167 private final SelectRelationAction selectRelationAction = new SelectRelationAction(false);
168
169 /**
170 * The Add button (needed to be able to disable it)
171 */
172 private final SideButton btnAdd = new SideButton(addAction);
173 /**
174 * The Edit button (needed to be able to disable it)
175 */
176 private final SideButton btnEdit = new SideButton(editAction);
177 /**
178 * The Delete button (needed to be able to disable it)
179 */
180 private final SideButton btnDel = new SideButton(deleteAction);
181 /**
182 * Matching preset display class
183 */
184 private final PresetListPanel presets = new PresetListPanel();
185
186 /**
187 * Text to display when nothing selected.
188 */
189 private final JLabel selectSth = new JLabel("<html><p>"
190 + tr("Select objects for which to change properties.") + "</p></html>");
191
192 private PresetHandler presetHandler = new PresetHandler() {
193 @Override public void updateTags(List<Tag> tags) {
194 Command command = TaggingPreset.createCommand(getSelection(), tags);
195 if (command != null) Main.main.undoRedo.add(command);
196 }
197
198 @Override public Collection<OsmPrimitive> getSelection() {
199 if (Main.main == null) return null;
200 if (Main.main.getCurrentDataSet() == null) return null;
201 return Main.main.getCurrentDataSet().getSelected();
202 }
203 };
204
205 // <editor-fold defaultstate="collapsed" desc="Dialog construction and helper methods">
206
207 /**
208 * Create a new PropertiesDialog
209 * @param mapFrame The parent map fram
210 */
211 public PropertiesDialog(MapFrame mapFrame) {
212 super(tr("Properties/Memberships"), "propertiesdialog", tr("Properties for selected objects."),
213 Shortcut.registerShortcut("subwindow:properties", tr("Toggle: {0}", tr("Properties/Memberships")), KeyEvent.VK_P,
214 Shortcut.ALT_SHIFT), 150, true);
215
216 setupPropertiesMenu();
217 buildPropertiesTable();
218
219 setupMembershipMenu();
220 buildMembershipTable();
221
222 // combine both tables and wrap them in a scrollPane
223 JPanel bothTables = new JPanel();
224 boolean top = Main.pref.getBoolean("properties.presets.top", true);
225 bothTables.setLayout(new GridBagLayout());
226 if(top) {
227 bothTables.add(presets, GBC.std().fill(GBC.HORIZONTAL).insets(5, 2, 5, 2).anchor(GBC.NORTHWEST));
228 double epsilon = Double.MIN_VALUE; // need to set a weight or else anchor value is ignored
229 bothTables.add(pluginHook, GBC.eol().insets(0,1,1,1).anchor(GBC.NORTHEAST).weight(epsilon, epsilon));
230 }
231 bothTables.add(selectSth, GBC.eol().fill().insets(10, 10, 10, 10));
232 bothTables.add(propertyTable.getTableHeader(), GBC.eol().fill(GBC.HORIZONTAL));
233 bothTables.add(propertyTable, GBC.eol().fill(GBC.BOTH));
234 bothTables.add(membershipTable.getTableHeader(), GBC.eol().fill(GBC.HORIZONTAL));
235 bothTables.add(membershipTable, GBC.eol().fill(GBC.BOTH));
236 if(!top) {
237 bothTables.add(presets, GBC.eol().fill(GBC.HORIZONTAL).insets(5, 2, 5, 2));
238 }
239
240 setupKeyboardShortcuts();
241
242 // Let the action know when selection in the tables change
243 propertyTable.getSelectionModel().addListSelectionListener(editAction);
244 membershipTable.getSelectionModel().addListSelectionListener(editAction);
245 propertyTable.getSelectionModel().addListSelectionListener(deleteAction);
246 membershipTable.getSelectionModel().addListSelectionListener(deleteAction);
247
248
249 JScrollPane scrollPane = (JScrollPane) createLayout(bothTables, true, Arrays.asList(new SideButton[] {
250 this.btnAdd, this.btnEdit, this.btnDel
251 }));
252
253 MouseClickWatch mouseClickWatch = new MouseClickWatch();
254 propertyTable.addMouseListener(mouseClickWatch);
255 membershipTable.addMouseListener(mouseClickWatch);
256 scrollPane.addMouseListener(mouseClickWatch);
257
258 selectSth.setPreferredSize(scrollPane.getSize());
259 presets.setSize(scrollPane.getSize());
260
261 editHelper.loadTagsIfNeeded();
262 }
263
264 private void buildPropertiesTable() {
265 // setting up the properties table
266
267 propertyData.setColumnIdentifiers(new String[]{tr("Key"),tr("Value")});
268 propertyTable.setSelectionMode(ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
269 propertyTable.getTableHeader().setReorderingAllowed(false);
270
271 propertyTable.getColumnModel().getColumn(1).setCellRenderer(new DefaultTableCellRenderer(){
272 @Override public Component getTableCellRendererComponent(JTable table, Object value,
273 boolean isSelected, boolean hasFocus, int row, int column) {
274 Component c = super.getTableCellRendererComponent(table, value, isSelected, false, row, column);
275 if (value == null)
276 return this;
277 if (c instanceof JLabel) {
278 String str = null;
279 if (value instanceof String) {
280 str = (String) value;
281 } else if (value instanceof Map<?, ?>) {
282 Map<?, ?> v = (Map<?, ?>) value;
283 if (v.size() != 1) {
284 str=tr("<different>");
285 c.setFont(c.getFont().deriveFont(Font.ITALIC));
286 } else {
287 final Map.Entry<?, ?> entry = v.entrySet().iterator().next();
288 str = (String) entry.getKey();
289 }
290 }
291 ((JLabel)c).setText(str);
292 }
293 return c;
294 }
295 });
296 }
297
298 private void buildMembershipTable() {
299 membershipData.setColumnIdentifiers(new String[]{tr("Member Of"),tr("Role"),tr("Position")});
300 membershipTable.setSelectionMode(ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
301
302 TableColumnModel mod = membershipTable.getColumnModel();
303 membershipTable.getTableHeader().setReorderingAllowed(false);
304 mod.getColumn(0).setCellRenderer(new DefaultTableCellRenderer() {
305 @Override public Component getTableCellRendererComponent(JTable table, Object value,
306 boolean isSelected, boolean hasFocus, int row, int column) {
307 Component c = super.getTableCellRendererComponent(table, value, isSelected, false, row, column);
308 if (value == null)
309 return this;
310 if (c instanceof JLabel) {
311 JLabel label = (JLabel)c;
312 Relation r = (Relation)value;
313 label.setText(r.getDisplayName(DefaultNameFormatter.getInstance()));
314 if (r.isDisabledAndHidden()) {
315 label.setFont(label.getFont().deriveFont(Font.ITALIC));
316 }
317 }
318 return c;
319 }
320 });
321
322 mod.getColumn(1).setCellRenderer(new DefaultTableCellRenderer() {
323 @Override public Component getTableCellRendererComponent(JTable table, Object value,
324 boolean isSelected, boolean hasFocus, int row, int column) {
325 if (value == null)
326 return this;
327 Component c = super.getTableCellRendererComponent(table, value, isSelected, false, row, column);
328 boolean isDisabledAndHidden = (((Relation)table.getValueAt(row, 0))).isDisabledAndHidden();
329 if (c instanceof JLabel) {
330 JLabel label = (JLabel)c;
331 MemberInfo col = (MemberInfo) value;
332
333 String text = null;
334 for (RelationMember r : col.role) {
335 if (text == null) {
336 text = r.getRole();
337 }
338 else if (!text.equals(r.getRole())) {
339 text = tr("<different>");
340 break;
341 }
342 }
343
344 label.setText(text);
345 if (isDisabledAndHidden) {
346 label.setFont(label.getFont().deriveFont(Font.ITALIC));
347 }
348 }
349 return c;
350 }
351 });
352
353 mod.getColumn(2).setCellRenderer(new DefaultTableCellRenderer() {
354 @Override public Component getTableCellRendererComponent(JTable table, Object value,
355 boolean isSelected, boolean hasFocus, int row, int column) {
356 Component c = super.getTableCellRendererComponent(table, value, isSelected, false, row, column);
357 boolean isDisabledAndHidden = (((Relation)table.getValueAt(row, 0))).isDisabledAndHidden();
358 if (c instanceof JLabel) {
359 JLabel label = (JLabel)c;
360 label.setText(((MemberInfo) table.getValueAt(row, 1)).getPositionString());
361 if (isDisabledAndHidden) {
362 label.setFont(label.getFont().deriveFont(Font.ITALIC));
363 }
364 }
365 return c;
366 }
367 });
368 mod.getColumn(2).setPreferredWidth(20);
369 mod.getColumn(1).setPreferredWidth(40);
370 mod.getColumn(0).setPreferredWidth(200);
371 }
372
373 /**
374 * creates the popup menu @field membershipMenu and its launcher on membership table
375 */
376 private void setupMembershipMenu() {
377 // setting up the membership table
378 membershipMenuHandler.addAction(addRelationToSelectionAction);
379 membershipMenuHandler.addAction(selectRelationAction);
380 membershipMenuHandler.addAction(addMembersToSelectionAction);
381 membershipMenuHandler.addAction(downloadSelectedIncompleteMembersAction);
382 membershipMenu.addSeparator();
383 membershipMenu.add(helpAction);
384
385 membershipTable.addMouseListener(new PopupMenuLauncher() {
386 @Override
387 public void launch(MouseEvent evt) {
388 Point p = evt.getPoint();
389 int row = membershipTable.rowAtPoint(p);
390 int idx[] = membershipTable.getSelectedRows();
391 // if nothing or one row is selected, select row under mouse instead
392 if (idx.length<2 && row>-1) {
393 membershipTable.changeSelection(row, 0, false, false);
394 idx = new int[]{row};
395 }
396 List<Relation> rels = new ArrayList<Relation>();
397 for (int i: idx) {
398 Relation r = (Relation) (membershipData.getValueAt(i, 0));
399 rels.add(r);
400 }
401 membershipMenuHandler.setPrimitives(rels);
402 membershipMenu.show(membershipTable, p.x, p.y-3);
403 }
404 });
405 }
406
407 /**
408 * creates the popup menu @field propertyMenu and its launcher on property table
409 */
410 private void setupPropertiesMenu() {
411 propertyMenu.add(pasteValueAction);
412 propertyMenu.add(copyValueAction);
413 propertyMenu.add(copyKeyValueAction);
414 propertyMenu.add(copyAllKeyValueAction);
415 propertyMenu.addSeparator();
416 propertyMenu.add(searchActionAny);
417 propertyMenu.add(searchActionSame);
418 propertyMenu.addSeparator();
419 propertyMenu.add(helpAction);
420 propertyTable.addMouseListener(new PopupMenuLauncher() {
421 @Override
422 public void launch(MouseEvent evt) {
423 Point p = evt.getPoint();
424 int row = propertyTable.rowAtPoint(p);
425 int selectedCount = propertyTable.getSelectedRowCount();
426 // if nothing or one row is selected, select row under mouse instead
427 if (selectedCount<2 && row>-1) {
428 propertyTable.changeSelection(row, 0, false, false);
429 }
430 if (selectedCount>=2 || row>-1) {
431 propertyMenu.show(propertyTable, p.x, p.y-3);
432 }
433 }
434 });
435 }
436
437 /**
438 * Assignas all needed keys like Enter and Spacebar to most important actions
439 */
440 private void setupKeyboardShortcuts() {
441
442 // ENTER = editAction, open "edit" dialog
443 propertyTable.getInputMap(JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT)
444 .put(KeyStroke.getKeyStroke(KeyEvent.VK_ENTER, 0),"onTableEnter");
445 propertyTable.getActionMap().put("onTableEnter",editAction);
446 membershipTable.getInputMap(JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT)
447 .put(KeyStroke.getKeyStroke(KeyEvent.VK_ENTER, 0),"onTableEnter");
448 membershipTable.getActionMap().put("onTableEnter",editAction);
449
450 // INSERT button = addAction, open "add property" dialog
451 propertyTable.getInputMap(JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT)
452 .put(KeyStroke.getKeyStroke(KeyEvent.VK_INSERT, 0),"onTableInsert");
453 propertyTable.getActionMap().put("onTableInsert",addAction);
454
455 // unassign some standard shortcuts for JTable to allow upload / download
456 InputMapUtils.unassignCtrlShiftUpDown(propertyTable, JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT);
457 // allow using enter to add tags for all look&feel configurations
458 InputMapUtils.enableEnter(this.btnAdd);
459
460 // DEL button = deleteAction
461 getInputMap(JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT).put(
462 KeyStroke.getKeyStroke(KeyEvent.VK_DELETE, 0),"delete"
463 );
464 getActionMap().put("delete", deleteAction);
465
466 // F1 button = custom help action
467 getInputMap(JComponent.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT).put(
468 KeyStroke.getKeyStroke(KeyEvent.VK_F1, 0), "onHelp");
469 getActionMap().put("onHelp", helpAction);
470 }
471
472 /**
473 * This simply fires up an {@link RelationEditor} for the relation shown; everything else
474 * is the editor's business.
475 *
476 * @param row
477 */
478 private void editMembership(int row) {
479 Relation relation = (Relation)membershipData.getValueAt(row, 0);
480 Main.map.relationListDialog.selectRelation(relation);
481 RelationEditor.getEditor(
482 Main.map.mapView.getEditLayer(),
483 relation,
484 ((MemberInfo) membershipData.getValueAt(row, 1)).role).setVisible(true);
485 }
486
487 private int findRow(TableModel model, Object value) {
488 for (int i=0; i<model.getRowCount(); i++) {
489 if (model.getValueAt(i, 0).equals(value))
490 return i;
491 }
492 return -1;
493 }
494
495 /**
496 * Update selection status, call @{link #selectionChanged} function.
497 */
498 private void updateSelection() {
499 if (Main.main.getCurrentDataSet() == null) {
500 selectionChanged(Collections.<OsmPrimitive>emptyList());
501 } else {
502 selectionChanged(Main.main.getCurrentDataSet().getSelected());
503 }
504 }
505
506 // </editor-fold>
507
508 // <editor-fold defaultstate="collapsed" desc="Event listeners methods">
509
510 @Override
511 public void showNotify() {
512 DatasetEventManager.getInstance().addDatasetListener(dataChangedAdapter, FireMode.IN_EDT_CONSOLIDATED);
513 SelectionEventManager.getInstance().addSelectionListener(this, FireMode.IN_EDT_CONSOLIDATED);
514 MapView.addEditLayerChangeListener(this);
515 for (JosmAction action : josmActions) {
516 Main.registerActionShortcut(action);
517 }
518 updateSelection();
519 }
520
521 @Override
522 public void hideNotify() {
523 DatasetEventManager.getInstance().removeDatasetListener(dataChangedAdapter);
524 SelectionEventManager.getInstance().removeSelectionListener(this);
525 MapView.removeEditLayerChangeListener(this);
526 for (JosmAction action : josmActions) {
527 Main.unregisterActionShortcut(action);
528 }
529 }
530
531 @Override
532 public void setVisible(boolean b) {
533 super.setVisible(b);
534 if (b && Main.main.getCurrentDataSet() != null) {
535 selectionChanged(Main.main.getCurrentDataSet().getSelected());
536 }
537 }
538
539 @Override
540 public void destroy() {
541 super.destroy();
542 for (JosmAction action : josmActions) {
543 action.destroy();
544 }
545 Container parent = pluginHook.getParent();
546 if (parent != null) {
547 parent.remove(pluginHook);
548 }
549 }
550
551 @Override
552 public void selectionChanged(Collection<? extends OsmPrimitive> newSelection) {
553 if (!isVisible())
554 return;
555 if (propertyTable == null)
556 return; // selection changed may be received in base class constructor before init
557 if (propertyTable.getCellEditor() != null) {
558 propertyTable.getCellEditor().cancelCellEditing();
559 }
560
561 String selectedTag;
562 Relation selectedRelation = null;
563 selectedTag = editHelper.getChangedKey(); // select last added or last edited key by default
564 if (selectedTag == null && propertyTable.getSelectedRowCount() == 1) {
565 selectedTag = (String)propertyData.getValueAt(propertyTable.getSelectedRow(), 0);
566 }
567 if (membershipTable.getSelectedRowCount() == 1) {
568 selectedRelation = (Relation)membershipData.getValueAt(membershipTable.getSelectedRow(), 0);
569 }
570
571 // re-load property data
572 propertyData.setRowCount(0);
573
574 final Map<String, Integer> keyCount = new HashMap<String, Integer>();
575 final Map<String, String> tags = new HashMap<String, String>();
576 valueCount.clear();
577 EnumSet<PresetType> types = EnumSet.noneOf(TaggingPreset.PresetType.class);
578 for (OsmPrimitive osm : newSelection) {
579 types.add(PresetType.forPrimitive(osm));
580 for (String key : osm.keySet()) {
581 String value = osm.get(key);
582 keyCount.put(key, keyCount.containsKey(key) ? keyCount.get(key) + 1 : 1);
583 if (valueCount.containsKey(key)) {
584 Map<String, Integer> v = valueCount.get(key);
585 v.put(value, v.containsKey(value) ? v.get(value) + 1 : 1);
586 } else {
587 TreeMap<String, Integer> v = new TreeMap<String, Integer>();
588 v.put(value, 1);
589 valueCount.put(key, v);
590 }
591 }
592 }
593 for (Entry<String, Map<String, Integer>> e : valueCount.entrySet()) {
594 int count = 0;
595 for (Entry<String, Integer> e1 : e.getValue().entrySet()) {
596 count += e1.getValue();
597 }
598 if (count < newSelection.size()) {
599 e.getValue().put("", newSelection.size() - count);
600 }
601 propertyData.addRow(new Object[]{e.getKey(), e.getValue()});
602 tags.put(e.getKey(), e.getValue().size() == 1
603 ? e.getValue().keySet().iterator().next() : tr("<different>"));
604 }
605
606 membershipData.setRowCount(0);
607
608 Map<Relation, MemberInfo> roles = new HashMap<Relation, MemberInfo>();
609 for (OsmPrimitive primitive: newSelection) {
610 for (OsmPrimitive ref: primitive.getReferrers()) {
611 if (ref instanceof Relation && !ref.isIncomplete() && !ref.isDeleted()) {
612 Relation r = (Relation) ref;
613 MemberInfo mi = roles.get(r);
614 if(mi == null) {
615 mi = new MemberInfo();
616 }
617 roles.put(r, mi);
618 int i = 1;
619 for (RelationMember m : r.getMembers()) {
620 if (m.getMember() == primitive) {
621 mi.add(m, i);
622 }
623 ++i;
624 }
625 }
626 }
627 }
628
629 List<Relation> sortedRelations = new ArrayList<Relation>(roles.keySet());
630 Collections.sort(sortedRelations, new Comparator<Relation>() {
631 @Override public int compare(Relation o1, Relation o2) {
632 int comp = Boolean.valueOf(o1.isDisabledAndHidden()).compareTo(o2.isDisabledAndHidden());
633 if (comp == 0) {
634 comp = o1.getDisplayName(DefaultNameFormatter.getInstance()).compareTo(o2.getDisplayName(DefaultNameFormatter.getInstance()));
635 }
636 return comp;
637 }}
638 );
639
640 for (Relation r: sortedRelations) {
641 membershipData.addRow(new Object[]{r, roles.get(r)});
642 }
643
644 presets.updatePresets(types, tags, presetHandler);
645
646 membershipTable.getTableHeader().setVisible(membershipData.getRowCount() > 0);
647 membershipTable.setVisible(membershipData.getRowCount() > 0);
648
649 boolean hasSelection = !newSelection.isEmpty();
650 boolean hasTags = hasSelection && propertyData.getRowCount() > 0;
651 boolean hasMemberships = hasSelection && membershipData.getRowCount() > 0;
652 btnAdd.setEnabled(hasSelection);
653 btnEdit.setEnabled(hasTags || hasMemberships);
654 btnDel.setEnabled(hasTags || hasMemberships);
655 propertyTable.setVisible(hasTags);
656 propertyTable.getTableHeader().setVisible(hasTags);
657 selectSth.setVisible(!hasSelection);
658 pluginHook.setVisible(hasSelection);
659
660 int selectedIndex;
661 if (selectedTag != null && (selectedIndex = findRow(propertyData, selectedTag)) != -1) {
662 propertyTable.changeSelection(selectedIndex, 0, false, false);
663 } else if (selectedRelation != null && (selectedIndex = findRow(membershipData, selectedRelation)) != -1) {
664 membershipTable.changeSelection(selectedIndex, 0, false, false);
665 } else if(hasTags) {
666 propertyTable.changeSelection(0, 0, false, false);
667 } else if(hasMemberships) {
668 membershipTable.changeSelection(0, 0, false, false);
669 }
670
671 if(propertyData.getRowCount() != 0 || membershipData.getRowCount() != 0) {
672 setTitle(tr("Properties: {0} / Memberships: {1}",
673 propertyData.getRowCount(), membershipData.getRowCount()));
674 } else {
675 setTitle(tr("Properties / Memberships"));
676 }
677 }
678
679 /* ---------------------------------------------------------------------------------- */
680 /* EditLayerChangeListener */
681 /* ---------------------------------------------------------------------------------- */
682 @Override
683 public void editLayerChanged(OsmDataLayer oldLayer, OsmDataLayer newLayer) {
684 if (newLayer == null) editHelper.saveTagsIfNeeded();
685 // it is time to save history of tags
686
687 updateSelection();
688 }
689
690 @Override
691 public void processDatasetEvent(AbstractDatasetChangedEvent event) {
692 updateSelection();
693 }
694
695 // </editor-fold>
696
697 // <editor-fold defaultstate="collapsed" desc="Methods that are called by plugins to extend fuctionalty ">
698
699 /**
700 * Replies the property popup menu handler.
701 * @return The property popup menu handler
702 */
703 public PopupMenuHandler getPropertyPopupMenuHandler() {
704 return propertyMenuHandler;
705 }
706
707 @SuppressWarnings("unchecked")
708 public Tag getSelectedProperty() {
709 int row = propertyTable.getSelectedRow();
710 if (row == -1) return null;
711 TreeMap<String, Integer> map = (TreeMap<String, Integer>) propertyData.getValueAt(row, 1);
712 return new Tag(
713 propertyData.getValueAt(row, 0).toString(),
714 map.size() > 1 ? "" : map.keySet().iterator().next());
715 }
716
717 /**
718 * Replies the membership popup menu handler.
719 * @return The membership popup menu handler
720 */
721 public PopupMenuHandler getMembershipPopupMenuHandler() {
722 return membershipMenuHandler;
723 }
724
725 public IRelation getSelectedMembershipRelation() {
726 int row = membershipTable.getSelectedRow();
727 return row > -1 ? (IRelation) membershipData.getValueAt(row, 0) : null;
728 }
729
730 // </editor-fold>
731
732 /**
733 * Class that watches for mouse clicks
734 * @author imi
735 */
736 public class MouseClickWatch extends MouseAdapter {
737 @Override public void mouseClicked(MouseEvent e) {
738 if (e.getClickCount() < 2)
739 {
740 // single click, clear selection in other table not clicked in
741 if (e.getSource() == propertyTable) {
742 membershipTable.clearSelection();
743 } else if (e.getSource() == membershipTable) {
744 propertyTable.clearSelection();
745 }
746 }
747 // double click, edit or add property
748 else if (e.getSource() == propertyTable)
749 {
750 int row = propertyTable.rowAtPoint(e.getPoint());
751 if (row > -1) {
752 boolean focusOnKey = (propertyTable.columnAtPoint(e.getPoint()) == 0);
753 editHelper.editProperty(row, focusOnKey);
754 } else {
755 editHelper.addProperty();
756 btnAdd.requestFocusInWindow();
757 }
758 } else if (e.getSource() == membershipTable) {
759 int row = membershipTable.rowAtPoint(e.getPoint());
760 if (row > -1) {
761 editMembership(row);
762 }
763 }
764 else
765 {
766 editHelper.addProperty();
767 btnAdd.requestFocusInWindow();
768 }
769 }
770 @Override public void mousePressed(MouseEvent e) {
771 if (e.getSource() == propertyTable) {
772 membershipTable.clearSelection();
773 } else if (e.getSource() == membershipTable) {
774 propertyTable.clearSelection();
775 }
776 }
777
778 }
779
780 static class MemberInfo {
781 List<RelationMember> role = new ArrayList<RelationMember>();
782 List<Integer> position = new ArrayList<Integer>();
783 private String positionString = null;
784 void add(RelationMember r, Integer p) {
785 role.add(r);
786 position.add(p);
787 }
788 String getPositionString() {
789 if (positionString == null) {
790 Collections.sort(position);
791 positionString = String.valueOf(position.get(0));
792 int cnt = 0;
793 int last = position.get(0);
794 for (int i = 1; i < position.size(); ++i) {
795 int cur = position.get(i);
796 if (cur == last + 1) {
797 ++cnt;
798 } else if (cnt == 0) {
799 positionString += "," + String.valueOf(cur);
800 } else {
801 positionString += "-" + String.valueOf(last);
802 positionString += "," + String.valueOf(cur);
803 cnt = 0;
804 }
805 last = cur;
806 }
807 if (cnt >= 1) {
808 positionString += "-" + String.valueOf(last);
809 }
810 }
811 if (positionString.length() > 20) {
812 positionString = positionString.substring(0, 17) + "...";
813 }
814 return positionString;
815 }
816 }
817
818 /**
819 * Class that allows fast creation of read-only table model with String columns
820 */
821 public static class ReadOnlyTableModel extends DefaultTableModel {
822 @Override public boolean isCellEditable(int row, int column) {
823 return false;
824 }
825 @Override public Class<?> getColumnClass(int columnIndex) {
826 return String.class;
827 }
828 };
829
830 /**
831 * Action handling delete button press in properties dialog.
832 */
833 class DeleteAction extends JosmAction implements ListSelectionListener {
834
835 public DeleteAction() {
836 super(tr("Delete"), "dialogs/delete", tr("Delete the selected key in all objects"),
837 Shortcut.registerShortcut("properties:delete", tr("Delete Properties"), KeyEvent.VK_D,
838 Shortcut.ALT_CTRL_SHIFT), false);
839 updateEnabledState();
840 }
841
842 protected void deleteProperties(int[] rows){
843 // convert list of rows to HashMap (and find gap for nextKey)
844 HashMap<String, String> tags = new HashMap<String, String>(rows.length);
845 int nextKeyIndex = rows[0];
846 for (int row : rows) {
847 String key = propertyData.getValueAt(row, 0).toString();
848 if (row == nextKeyIndex + 1) {
849 nextKeyIndex = row; // no gap yet
850 }
851 tags.put(key, null);
852 }
853
854 // find key to select after deleting other properties
855 String nextKey = null;
856 int rowCount = propertyData.getRowCount();
857 if (rowCount > rows.length) {
858 if (nextKeyIndex == rows[rows.length-1]) {
859 // no gap found, pick next or previous key in list
860 nextKeyIndex = (nextKeyIndex + 1 < rowCount ? nextKeyIndex + 1 : rows[0] - 1);
861 } else {
862 // gap found
863 nextKeyIndex++;
864 }
865 nextKey = (String)propertyData.getValueAt(nextKeyIndex, 0);
866 }
867
868 Collection<OsmPrimitive> sel = Main.main.getCurrentDataSet().getSelected();
869 Main.main.undoRedo.add(new ChangePropertyCommand(sel, tags));
870
871 membershipTable.clearSelection();
872 if (nextKey != null) {
873 propertyTable.changeSelection(findRow(propertyData, nextKey), 0, false, false);
874 }
875 }
876
877 protected void deleteFromRelation(int row) {
878 Relation cur = (Relation)membershipData.getValueAt(row, 0);
879
880 Relation nextRelation = null;
881 int rowCount = membershipTable.getRowCount();
882 if (rowCount > 1) {
883 nextRelation = (Relation)membershipData.getValueAt((row + 1 < rowCount ? row + 1 : row - 1), 0);
884 }
885
886 ExtendedDialog ed = new ExtendedDialog(Main.parent,
887 tr("Change relation"),
888 new String[] {tr("Delete from relation"), tr("Cancel")});
889 ed.setButtonIcons(new String[] {"dialogs/delete.png", "cancel.png"});
890 ed.setContent(tr("Really delete selection from relation {0}?", cur.getDisplayName(DefaultNameFormatter.getInstance())));
891 ed.toggleEnable("delete_from_relation");
892 ed.showDialog();
893
894 if(ed.getValue() != 1)
895 return;
896
897 Relation rel = new Relation(cur);
898 Collection<OsmPrimitive> sel = Main.main.getCurrentDataSet().getSelected();
899 for (OsmPrimitive primitive: sel) {
900 rel.removeMembersFor(primitive);
901 }
902 Main.main.undoRedo.add(new ChangeCommand(cur, rel));
903
904 propertyTable.clearSelection();
905 if (nextRelation != null) {
906 membershipTable.changeSelection(findRow(membershipData, nextRelation), 0, false, false);
907 }
908 }
909
910 @Override
911 public void actionPerformed(ActionEvent e) {
912 if (propertyTable.getSelectedRowCount() > 0) {
913 int[] rows = propertyTable.getSelectedRows();
914 deleteProperties(rows);
915 } else if (membershipTable.getSelectedRowCount() > 0) {
916 int[] rows = membershipTable.getSelectedRows();
917 // delete from last relation to convserve row numbers in the table
918 for (int i=rows.length-1; i>=0; i--) {
919 deleteFromRelation(rows[i]);
920 }
921 }
922 }
923
924 @Override
925 protected void updateEnabledState() {
926 setEnabled(
927 (propertyTable != null && propertyTable.getSelectedRowCount() >= 1)
928 || (membershipTable != null && membershipTable.getSelectedRowCount() > 0)
929 );
930 }
931
932 @Override
933 public void valueChanged(ListSelectionEvent e) {
934 updateEnabledState();
935 }
936 }
937
938 /**
939 * Action handling add button press in properties dialog.
940 */
941 class AddAction extends JosmAction {
942 public AddAction() {
943 super(tr("Add"), "dialogs/add", tr("Add a new key/value pair to all objects"),
944 Shortcut.registerShortcut("properties:add", tr("Add Property"), KeyEvent.VK_A,
945 Shortcut.ALT), false);
946 }
947
948 @Override
949 public void actionPerformed(ActionEvent e) {
950 editHelper.addProperty();
951 btnAdd.requestFocusInWindow();
952 }
953 }
954
955 /**
956 * Action handling edit button press in properties dialog.
957 */
958 class EditAction extends JosmAction implements ListSelectionListener {
959 public EditAction() {
960 super(tr("Edit"), "dialogs/edit", tr("Edit the value of the selected key for all objects"),
961 Shortcut.registerShortcut("properties:edit", tr("Edit Properties"), KeyEvent.VK_S,
962 Shortcut.ALT), false);
963 updateEnabledState();
964 }
965
966 @Override
967 public void actionPerformed(ActionEvent e) {
968 if (!isEnabled())
969 return;
970 if (propertyTable.getSelectedRowCount() == 1) {
971 int row = propertyTable.getSelectedRow();
972 editHelper.editProperty(row, false);
973 } else if (membershipTable.getSelectedRowCount() == 1) {
974 int row = membershipTable.getSelectedRow();
975 editMembership(row);
976 }
977 }
978
979 @Override
980 protected void updateEnabledState() {
981 setEnabled(
982 (propertyTable != null && propertyTable.getSelectedRowCount() == 1)
983 ^ (membershipTable != null && membershipTable.getSelectedRowCount() == 1)
984 );
985 }
986
987 @Override
988 public void valueChanged(ListSelectionEvent e) {
989 updateEnabledState();
990 }
991 }
992
993 class HelpAction extends AbstractAction {
994 public HelpAction() {
995 putValue(NAME, tr("Go to OSM wiki for tag help (F1)"));
996 putValue(SHORT_DESCRIPTION, tr("Launch browser with wiki help for selected object"));
997 putValue(SMALL_ICON, ImageProvider.get("dialogs", "search"));
998 }
999
1000 @Override
1001 public void actionPerformed(ActionEvent e) {
1002 try {
1003 String base = Main.pref.get("url.openstreetmap-wiki", "http://wiki.openstreetmap.org/wiki/");
1004 String lang = LanguageInfo.getWikiLanguagePrefix();
1005 final List<URI> uris = new ArrayList<URI>();
1006 int row;
1007 if (propertyTable.getSelectedRowCount() == 1) {
1008 row = propertyTable.getSelectedRow();
1009 String key = URLEncoder.encode(propertyData.getValueAt(row, 0).toString(), "UTF-8");
1010 @SuppressWarnings("unchecked")
1011 Map<String, Integer> m = (Map<String, Integer>) propertyData.getValueAt(row, 1);
1012 String val = URLEncoder.encode(m.entrySet().iterator().next().getKey(), "UTF-8");
1013
1014 uris.add(new URI(String.format("%s%sTag:%s=%s", base, lang, key, val)));
1015 uris.add(new URI(String.format("%sTag:%s=%s", base, key, val)));
1016 uris.add(new URI(String.format("%s%sKey:%s", base, lang, key)));
1017 uris.add(new URI(String.format("%sKey:%s", base, key)));
1018 uris.add(new URI(String.format("%s%sMap_Features", base, lang)));
1019 uris.add(new URI(String.format("%sMap_Features", base)));
1020 } else if (membershipTable.getSelectedRowCount() == 1) {
1021 row = membershipTable.getSelectedRow();
1022 String type = URLEncoder.encode(
1023 ((Relation)membershipData.getValueAt(row, 0)).get("type"), "UTF-8"
1024 );
1025
1026 if (type != null && !type.equals("")) {
1027 uris.add(new URI(String.format("%s%sRelation:%s", base, lang, type)));
1028 uris.add(new URI(String.format("%sRelation:%s", base, type)));
1029 }
1030
1031 uris.add(new URI(String.format("%s%sRelations", base, lang)));
1032 uris.add(new URI(String.format("%sRelations", base)));
1033 } else {
1034 // give the generic help page, if more than one element is selected
1035 uris.add(new URI(String.format("%s%sMap_Features", base, lang)));
1036 uris.add(new URI(String.format("%sMap_Features", base)));
1037 }
1038
1039 Main.worker.execute(new Runnable(){
1040 @Override public void run() {
1041 try {
1042 // find a page that actually exists in the wiki
1043 HttpURLConnection conn;
1044 for (URI u : uris) {
1045 conn = Utils.openHttpConnection(u.toURL());
1046 conn.setConnectTimeout(Main.pref.getInteger("socket.timeout.connect",15)*1000);
1047
1048 if (conn.getResponseCode() != 200) {
1049 Main.info("INFO: {0} does not exist", u);
1050 conn.disconnect();
1051 } else {
1052 int osize = conn.getContentLength();
1053 conn.disconnect();
1054
1055 conn = Utils.openHttpConnection(new URI(u.toString()
1056 .replace("=", "%3D") /* do not URLencode whole string! */
1057 .replaceFirst("/wiki/", "/w/index.php?redirect=no&title=")
1058 ).toURL());
1059 conn.setConnectTimeout(Main.pref.getInteger("socket.timeout.connect",15)*1000);
1060
1061 /* redirect pages have different content length, but retrieving a "nonredirect"
1062 * page using index.php and the direct-link method gives slightly different
1063 * content lengths, so we have to be fuzzy.. (this is UGLY, recode if u know better)
1064 */
1065 if (Math.abs(conn.getContentLength() - osize) > 200) {
1066 Main.info("INFO: {0} is a mediawiki redirect", u);
1067 conn.disconnect();
1068 } else {
1069 Main.info("INFO: browsing to {0}", u);
1070 conn.disconnect();
1071
1072 OpenBrowser.displayUrl(u.toString());
1073 break;
1074 }
1075 }
1076 }
1077 } catch (Exception e) {
1078 e.printStackTrace();
1079 }
1080 }
1081 });
1082 } catch (Exception e1) {
1083 e1.printStackTrace();
1084 }
1085 }
1086 }
1087
1088 class PasteValueAction extends AbstractAction {
1089 public PasteValueAction() {
1090 putValue(NAME, tr("Paste Value"));
1091 putValue(SHORT_DESCRIPTION, tr("Paste the value of the selected tag from clipboard"));
1092 }
1093
1094 @Override
1095 public void actionPerformed(ActionEvent ae) {
1096 if (propertyTable.getSelectedRowCount() != 1)
1097 return;
1098 String key = propertyData.getValueAt(propertyTable.getSelectedRow(), 0).toString();
1099 Collection<OsmPrimitive> sel = Main.main.getCurrentDataSet().getSelected();
1100 String clipboard = Utils.getClipboardContent();
1101 if (sel.isEmpty() || clipboard == null)
1102 return;
1103 Main.main.undoRedo.add(new ChangePropertyCommand(sel, key, Utils.strip(clipboard)));
1104 }
1105 }
1106
1107 abstract class AbstractCopyAction extends AbstractAction {
1108
1109 protected abstract Collection<String> getString(OsmPrimitive p, String key);
1110
1111 @Override
1112 public void actionPerformed(ActionEvent ae) {
1113 int rows[] = propertyTable.getSelectedRows();
1114 Set<String> values = new TreeSet<String>();
1115 Collection<OsmPrimitive> sel = Main.main.getCurrentDataSet().getSelected();
1116 if (rows.length == 0 || sel.isEmpty()) return;
1117
1118 for (int row: rows) {
1119 String key = propertyData.getValueAt(row, 0).toString();
1120 if (sel.isEmpty())
1121 return;
1122 for (OsmPrimitive p : sel) {
1123 Collection<String> s = getString(p,key);
1124 if (s != null) {
1125 values.addAll(s);
1126 }
1127 }
1128 }
1129 if (!values.isEmpty()) {
1130 Utils.copyToClipboard(Utils.join("\n", values));
1131 }
1132 }
1133 }
1134
1135 class CopyValueAction extends AbstractCopyAction {
1136
1137 public CopyValueAction() {
1138 putValue(NAME, tr("Copy Value"));
1139 putValue(SHORT_DESCRIPTION, tr("Copy the value of the selected tag to clipboard"));
1140 }
1141
1142 @Override
1143 protected Collection<String> getString(OsmPrimitive p, String key) {
1144 String v = p.get(key);
1145 return v == null ? null : Collections.singleton(v);
1146 }
1147 }
1148
1149 class CopyKeyValueAction extends AbstractCopyAction {
1150
1151 public CopyKeyValueAction() {
1152 putValue(NAME, tr("Copy Key/Value"));
1153 putValue(SHORT_DESCRIPTION, tr("Copy the key and value of the selected tag to clipboard"));
1154 }
1155
1156 @Override
1157 protected Collection<String> getString(OsmPrimitive p, String key) {
1158 String v = p.get(key);
1159 return v == null ? null : Collections.singleton(new Tag(key, v).toString());
1160 }
1161 }
1162
1163 class CopyAllKeyValueAction extends AbstractCopyAction {
1164
1165 public CopyAllKeyValueAction() {
1166 putValue(NAME, tr("Copy all Keys/Values"));
1167 putValue(SHORT_DESCRIPTION, tr("Copy the key and value of the all tags to clipboard"));
1168 }
1169
1170 @Override
1171 protected Collection<String> getString(OsmPrimitive p, String key) {
1172 List<String> r = new LinkedList<String>();
1173 for (Entry<String, String> kv : p.getKeys().entrySet()) {
1174 r.add(new Tag(kv.getKey(), kv.getValue()).toString());
1175 }
1176 return r;
1177 }
1178 }
1179
1180 class SearchAction extends AbstractAction {
1181 final boolean sameType;
1182
1183 public SearchAction(boolean sameType) {
1184 this.sameType = sameType;
1185 if (sameType) {
1186 putValue(NAME, tr("Search Key/Value/Type"));
1187 putValue(SHORT_DESCRIPTION, tr("Search with the key and value of the selected tag, restrict to type (i.e., node/way/relation)"));
1188 } else {
1189 putValue(NAME, tr("Search Key/Value"));
1190 putValue(SHORT_DESCRIPTION, tr("Search with the key and value of the selected tag"));
1191 }
1192 }
1193
1194 @Override
1195 public void actionPerformed(ActionEvent e) {
1196 if (propertyTable.getSelectedRowCount() != 1)
1197 return;
1198 String key = propertyData.getValueAt(propertyTable.getSelectedRow(), 0).toString();
1199 Collection<OsmPrimitive> sel = Main.main.getCurrentDataSet().getSelected();
1200 if (sel.isEmpty())
1201 return;
1202 String sep = "";
1203 String s = "";
1204 for (OsmPrimitive p : sel) {
1205 String val = p.get(key);
1206 if (val == null) {
1207 continue;
1208 }
1209 String t = "";
1210 if (!sameType) {
1211 t = "";
1212 } else if (p instanceof Node) {
1213 t = "type:node ";
1214 } else if (p instanceof Way) {
1215 t = "type:way ";
1216 } else if (p instanceof Relation) {
1217 t = "type:relation ";
1218 }
1219 s += sep + "(" + t + "\"" +
1220 org.openstreetmap.josm.actions.search.SearchAction.escapeStringForSearch(key) + "\"=\"" +
1221 org.openstreetmap.josm.actions.search.SearchAction.escapeStringForSearch(val) + "\")";
1222 sep = " OR ";
1223 }
1224
1225 SearchSetting ss = new SearchSetting(s, SearchMode.replace, true, false, false);
1226 org.openstreetmap.josm.actions.search.SearchAction.searchWithoutHistory(ss);
1227 }
1228 }
1229}
Note: See TracBrowser for help on using the repository browser.