source: josm/trunk/src/org/openstreetmap/josm/gui/history/HistoryBrowserModel.java@ 11322

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

fix #14032 - include editor (retrieved from created_by changeset tag) in history dialog

  • Property svn:eol-style set to native
File size: 31.3 KB
Line 
1// License: GPL. For details, see LICENSE file.
2package org.openstreetmap.josm.gui.history;
3
4import static org.openstreetmap.josm.tools.I18n.tr;
5
6import java.text.DateFormat;
7import java.util.ArrayList;
8import java.util.Collections;
9import java.util.HashSet;
10import java.util.List;
11import java.util.Set;
12
13import javax.swing.JTable;
14import javax.swing.table.AbstractTableModel;
15import javax.swing.table.TableModel;
16
17import org.openstreetmap.josm.Main;
18import org.openstreetmap.josm.data.osm.Changeset;
19import org.openstreetmap.josm.data.osm.Node;
20import org.openstreetmap.josm.data.osm.OsmPrimitive;
21import org.openstreetmap.josm.data.osm.OsmPrimitiveType;
22import org.openstreetmap.josm.data.osm.Relation;
23import org.openstreetmap.josm.data.osm.RelationMember;
24import org.openstreetmap.josm.data.osm.RelationMemberData;
25import org.openstreetmap.josm.data.osm.User;
26import org.openstreetmap.josm.data.osm.UserInfo;
27import org.openstreetmap.josm.data.osm.Way;
28import org.openstreetmap.josm.data.osm.event.AbstractDatasetChangedEvent;
29import org.openstreetmap.josm.data.osm.event.DataChangedEvent;
30import org.openstreetmap.josm.data.osm.event.DataSetListener;
31import org.openstreetmap.josm.data.osm.event.NodeMovedEvent;
32import org.openstreetmap.josm.data.osm.event.PrimitivesAddedEvent;
33import org.openstreetmap.josm.data.osm.event.PrimitivesRemovedEvent;
34import org.openstreetmap.josm.data.osm.event.RelationMembersChangedEvent;
35import org.openstreetmap.josm.data.osm.event.TagsChangedEvent;
36import org.openstreetmap.josm.data.osm.event.WayNodesChangedEvent;
37import org.openstreetmap.josm.data.osm.history.History;
38import org.openstreetmap.josm.data.osm.history.HistoryNode;
39import org.openstreetmap.josm.data.osm.history.HistoryOsmPrimitive;
40import org.openstreetmap.josm.data.osm.history.HistoryRelation;
41import org.openstreetmap.josm.data.osm.history.HistoryWay;
42import org.openstreetmap.josm.data.osm.visitor.AbstractVisitor;
43import org.openstreetmap.josm.gui.JosmUserIdentityManager;
44import org.openstreetmap.josm.gui.layer.Layer;
45import org.openstreetmap.josm.gui.layer.MainLayerManager.ActiveLayerChangeEvent;
46import org.openstreetmap.josm.gui.layer.MainLayerManager.ActiveLayerChangeListener;
47import org.openstreetmap.josm.gui.layer.OsmDataLayer;
48import org.openstreetmap.josm.gui.util.ChangeNotifier;
49import org.openstreetmap.josm.tools.CheckParameterUtil;
50import org.openstreetmap.josm.tools.date.DateUtils;
51
52/**
53 * This is the model used by the history browser.
54 *
55 * The model state consists of the following elements:
56 * <ul>
57 * <li>the {@link History} of a specific {@link OsmPrimitive}</li>
58 * <li>a dedicated version in this {@link History} called the {@link PointInTimeType#REFERENCE_POINT_IN_TIME}</li>
59 * <li>another version in this {@link History} called the {@link PointInTimeType#CURRENT_POINT_IN_TIME}</li>
60 * </ul>
61 * {@link HistoryBrowser} always compares the {@link PointInTimeType#REFERENCE_POINT_IN_TIME} with the
62 * {@link PointInTimeType#CURRENT_POINT_IN_TIME}.
63
64 * This model provides various {@link TableModel}s for {@link JTable}s used in {@link HistoryBrowser}, for
65 * instance:
66 * <ul>
67 * <li>{@link #getTagTableModel(PointInTimeType)} replies a {@link TableModel} for the tags of either of
68 * the two selected versions</li>
69 * <li>{@link #getNodeListTableModel(PointInTimeType)} replies a {@link TableModel} for the list of nodes of
70 * the two selected versions (if the current history provides information about a {@link Way}</li>
71 * <li> {@link #getRelationMemberTableModel(PointInTimeType)} replies a {@link TableModel} for the list of relation
72 * members of the two selected versions (if the current history provides information about a {@link Relation}</li>
73 * </ul>
74 *
75 * @see HistoryBrowser
76 */
77public class HistoryBrowserModel extends ChangeNotifier implements ActiveLayerChangeListener, DataSetListener {
78 /** the history of an OsmPrimitive */
79 private History history;
80 private HistoryOsmPrimitive reference;
81 private HistoryOsmPrimitive current;
82 /**
83 * latest isn't a reference of history. It's a clone of the currently edited
84 * {@link OsmPrimitive} in the current edit layer.
85 */
86 private HistoryOsmPrimitive latest;
87
88 private final VersionTableModel versionTableModel;
89 private final TagTableModel currentTagTableModel;
90 private final TagTableModel referenceTagTableModel;
91 private final DiffTableModel currentRelationMemberTableModel;
92 private final DiffTableModel referenceRelationMemberTableModel;
93 private final DiffTableModel referenceNodeListTableModel;
94 private final DiffTableModel currentNodeListTableModel;
95
96 /**
97 * constructor
98 */
99 public HistoryBrowserModel() {
100 versionTableModel = new VersionTableModel();
101 currentTagTableModel = new TagTableModel(PointInTimeType.CURRENT_POINT_IN_TIME);
102 referenceTagTableModel = new TagTableModel(PointInTimeType.REFERENCE_POINT_IN_TIME);
103 referenceNodeListTableModel = new DiffTableModel();
104 currentNodeListTableModel = new DiffTableModel();
105 currentRelationMemberTableModel = new DiffTableModel();
106 referenceRelationMemberTableModel = new DiffTableModel();
107
108 if (Main.main != null) {
109 OsmDataLayer editLayer = Main.getLayerManager().getEditLayer();
110 if (editLayer != null) {
111 editLayer.data.addDataSetListener(this);
112 }
113 }
114 Main.getLayerManager().addActiveLayerChangeListener(this);
115 }
116
117 /**
118 * Creates a new history browser model for a given history.
119 *
120 * @param history the history. Must not be null.
121 * @throws IllegalArgumentException if history is null
122 */
123 public HistoryBrowserModel(History history) {
124 this();
125 CheckParameterUtil.ensureParameterNotNull(history, "history");
126 setHistory(history);
127 }
128
129 /**
130 * replies the history managed by this model
131 * @return the history
132 */
133 public History getHistory() {
134 return history;
135 }
136
137 protected boolean canShowAsLatest(OsmPrimitive primitive) {
138 if (primitive == null)
139 return false;
140 if (primitive.isNew() || !primitive.isUsable())
141 return false;
142
143 //try creating a history primitive. if that fails, the primitive cannot be used.
144 try {
145 HistoryOsmPrimitive.forOsmPrimitive(primitive);
146 } catch (IllegalArgumentException ign) {
147 Main.trace(ign);
148 return false;
149 }
150
151 if (history == null)
152 return false;
153 // only show latest of the same version if it is modified
154 if (history.getByVersion(primitive.getVersion()) != null)
155 return primitive.isModified();
156
157 // if latest version from history is higher than a non existing primitive version,
158 // that means this version has been redacted and the primitive cannot be used.
159 if (history.getLatest().getVersion() > primitive.getVersion())
160 return false;
161
162 // latest has a higher version than one of the primitives
163 // in the history (probably because the history got out of sync
164 // with uploaded data) -> show the primitive as latest
165 return true;
166 }
167
168 /**
169 * sets the history to be managed by this model
170 *
171 * @param history the history
172 *
173 */
174 public void setHistory(History history) {
175 this.history = history;
176 if (history.getNumVersions() > 0) {
177 HistoryOsmPrimitive newLatest = null;
178 OsmDataLayer editLayer = Main.getLayerManager().getEditLayer();
179 if (editLayer != null) {
180 OsmPrimitive p = editLayer.data.getPrimitiveById(history.getId(), history.getType());
181 if (canShowAsLatest(p)) {
182 newLatest = new HistoryPrimitiveBuilder().build(p);
183 }
184 }
185 if (newLatest == null) {
186 current = history.getLatest();
187 int prevIndex = history.getNumVersions() - 2;
188 reference = prevIndex < 0 ? history.getEarliest() : history.get(prevIndex);
189 } else {
190 reference = history.getLatest();
191 current = newLatest;
192 }
193 setLatest(newLatest);
194 }
195 initTagTableModels();
196 fireModelChange();
197 }
198
199 protected void fireModelChange() {
200 initNodeListTableModels();
201 initMemberListTableModels();
202 fireStateChanged();
203 versionTableModel.fireTableDataChanged();
204 }
205
206 /**
207 * Replies the table model to be used in a {@link JTable} which
208 * shows the list of versions in this history.
209 *
210 * @return the table model
211 */
212 public VersionTableModel getVersionTableModel() {
213 return versionTableModel;
214 }
215
216 protected void initTagTableModels() {
217 currentTagTableModel.initKeyList();
218 referenceTagTableModel.initKeyList();
219 }
220
221 /**
222 * Should be called everytime either reference of current changes to update the diff.
223 * TODO: Maybe rename to reflect this? eg. updateNodeListTableModels
224 */
225 protected void initNodeListTableModels() {
226 if (current == null || current.getType() != OsmPrimitiveType.WAY
227 || reference == null || reference.getType() != OsmPrimitiveType.WAY)
228 return;
229 TwoColumnDiff diff = new TwoColumnDiff(
230 ((HistoryWay) reference).getNodes().toArray(),
231 ((HistoryWay) current).getNodes().toArray());
232 referenceNodeListTableModel.setRows(diff.referenceDiff, diff.referenceReversed);
233 currentNodeListTableModel.setRows(diff.currentDiff, false);
234 }
235
236 protected void initMemberListTableModels() {
237 if (current == null || current.getType() != OsmPrimitiveType.RELATION
238 || reference == null || reference.getType() != OsmPrimitiveType.RELATION)
239 return;
240 TwoColumnDiff diff = new TwoColumnDiff(
241 ((HistoryRelation) reference).getMembers().toArray(),
242 ((HistoryRelation) current).getMembers().toArray());
243 referenceRelationMemberTableModel.setRows(diff.referenceDiff, diff.referenceReversed);
244 currentRelationMemberTableModel.setRows(diff.currentDiff, false);
245 }
246
247 /**
248 * Replies the tag table model for the respective point in time.
249 *
250 * @param pointInTimeType the type of the point in time (must not be null)
251 * @return the tag table model
252 * @throws IllegalArgumentException if pointInTimeType is null
253 */
254 public TagTableModel getTagTableModel(PointInTimeType pointInTimeType) {
255 CheckParameterUtil.ensureParameterNotNull(pointInTimeType, "pointInTimeType");
256 if (pointInTimeType.equals(PointInTimeType.CURRENT_POINT_IN_TIME))
257 return currentTagTableModel;
258 else // REFERENCE_POINT_IN_TIME
259 return referenceTagTableModel;
260 }
261
262 /**
263 * Replies the node list table model for the respective point in time.
264 *
265 * @param pointInTimeType the type of the point in time (must not be null)
266 * @return the node list table model
267 * @throws IllegalArgumentException if pointInTimeType is null
268 */
269 public DiffTableModel getNodeListTableModel(PointInTimeType pointInTimeType) {
270 CheckParameterUtil.ensureParameterNotNull(pointInTimeType, "pointInTimeType");
271 if (pointInTimeType.equals(PointInTimeType.CURRENT_POINT_IN_TIME))
272 return currentNodeListTableModel;
273 else // REFERENCE_POINT_IN_TIME
274 return referenceNodeListTableModel;
275 }
276
277 /**
278 * Replies the relation member table model for the respective point in time.
279 *
280 * @param pointInTimeType the type of the point in time (must not be null)
281 * @return the relation member table model
282 * @throws IllegalArgumentException if pointInTimeType is null
283 */
284 public DiffTableModel getRelationMemberTableModel(PointInTimeType pointInTimeType) {
285 CheckParameterUtil.ensureParameterNotNull(pointInTimeType, "pointInTimeType");
286 if (pointInTimeType.equals(PointInTimeType.CURRENT_POINT_IN_TIME))
287 return currentRelationMemberTableModel;
288 else // REFERENCE_POINT_IN_TIME
289 return referenceRelationMemberTableModel;
290 }
291
292 /**
293 * Sets the {@link HistoryOsmPrimitive} which plays the role of a reference point
294 * in time (see {@link PointInTimeType}).
295 *
296 * @param reference the reference history primitive. Must not be null.
297 * @throws IllegalArgumentException if reference is null
298 * @throws IllegalStateException if this model isn't a assigned a history yet
299 * @throws IllegalArgumentException if reference isn't an history primitive for the history managed by this mode
300 *
301 * @see #setHistory(History)
302 * @see PointInTimeType
303 */
304 public void setReferencePointInTime(HistoryOsmPrimitive reference) {
305 CheckParameterUtil.ensureParameterNotNull(reference, "reference");
306 if (history == null)
307 throw new IllegalStateException(tr("History not initialized yet. Failed to set reference primitive."));
308 if (reference.getId() != history.getId())
309 throw new IllegalArgumentException(
310 tr("Failed to set reference. Reference ID {0} does not match history ID {1}.", reference.getId(), history.getId()));
311 HistoryOsmPrimitive primitive = history.getByVersion(reference.getVersion());
312 if (primitive == null)
313 throw new IllegalArgumentException(
314 tr("Failed to set reference. Reference version {0} not available in history.", reference.getVersion()));
315
316 this.reference = reference;
317 initTagTableModels();
318 initNodeListTableModels();
319 initMemberListTableModels();
320 fireStateChanged();
321 }
322
323 /**
324 * Sets the {@link HistoryOsmPrimitive} which plays the role of the current point
325 * in time (see {@link PointInTimeType}).
326 *
327 * @param current the reference history primitive. Must not be {@code null}.
328 * @throws IllegalArgumentException if reference is {@code null}
329 * @throws IllegalStateException if this model isn't a assigned a history yet
330 * @throws IllegalArgumentException if reference isn't an history primitive for the history managed by this mode
331 *
332 * @see #setHistory(History)
333 * @see PointInTimeType
334 */
335 public void setCurrentPointInTime(HistoryOsmPrimitive current) {
336 CheckParameterUtil.ensureParameterNotNull(current, "current");
337 if (history == null)
338 throw new IllegalStateException(tr("History not initialized yet. Failed to set current primitive."));
339 if (current.getId() != history.getId())
340 throw new IllegalArgumentException(
341 tr("Failed to set reference. Reference ID {0} does not match history ID {1}.", current.getId(), history.getId()));
342 HistoryOsmPrimitive primitive = history.getByVersion(current.getVersion());
343 if (primitive == null)
344 throw new IllegalArgumentException(
345 tr("Failed to set current primitive. Current version {0} not available in history.", current.getVersion()));
346 this.current = current;
347 initTagTableModels();
348 initNodeListTableModels();
349 initMemberListTableModels();
350 fireStateChanged();
351 }
352
353 /**
354 * Replies the history OSM primitive for the {@link PointInTimeType#CURRENT_POINT_IN_TIME}
355 *
356 * @return the history OSM primitive for the {@link PointInTimeType#CURRENT_POINT_IN_TIME} (may be null)
357 */
358 public HistoryOsmPrimitive getCurrentPointInTime() {
359 return getPointInTime(PointInTimeType.CURRENT_POINT_IN_TIME);
360 }
361
362 /**
363 * Replies the history OSM primitive for the {@link PointInTimeType#REFERENCE_POINT_IN_TIME}
364 *
365 * @return the history OSM primitive for the {@link PointInTimeType#REFERENCE_POINT_IN_TIME} (may be null)
366 */
367 public HistoryOsmPrimitive getReferencePointInTime() {
368 return getPointInTime(PointInTimeType.REFERENCE_POINT_IN_TIME);
369 }
370
371 /**
372 * replies the history OSM primitive for a given point in time
373 *
374 * @param type the type of the point in time (must not be null)
375 * @return the respective primitive. Can be null.
376 * @throws IllegalArgumentException if type is null
377 */
378 public HistoryOsmPrimitive getPointInTime(PointInTimeType type) {
379 CheckParameterUtil.ensureParameterNotNull(type, "type");
380 if (type.equals(PointInTimeType.CURRENT_POINT_IN_TIME))
381 return current;
382 else if (type.equals(PointInTimeType.REFERENCE_POINT_IN_TIME))
383 return reference;
384
385 // should not happen
386 return null;
387 }
388
389 /**
390 * Returns true if <code>primitive</code> is the latest primitive
391 * representing the version currently edited in the current data
392 * layer.
393 *
394 * @param primitive the primitive to check
395 * @return true if <code>primitive</code> is the latest primitive
396 */
397 public boolean isLatest(HistoryOsmPrimitive primitive) {
398 if (primitive == null)
399 return false;
400 return primitive == latest;
401 }
402
403 /**
404 * The table model for the list of versions in the current history
405 *
406 */
407 public final class VersionTableModel extends AbstractTableModel {
408
409 private VersionTableModel() {
410 }
411
412 @Override
413 public int getRowCount() {
414 if (history == null)
415 return 0;
416 int ret = history.getNumVersions();
417 if (latest != null) {
418 ret++;
419 }
420 return ret;
421 }
422
423 @Override
424 public Object getValueAt(int row, int column) {
425 switch (column) {
426 case VersionTableColumnModel.COL_VERSION:
427 return Long.toString(getPrimitive(row).getVersion());
428 case VersionTableColumnModel.COL_REFERENCE:
429 return isReferencePointInTime(row);
430 case VersionTableColumnModel.COL_CURRENT:
431 return isCurrentPointInTime(row);
432 case VersionTableColumnModel.COL_DATE:
433 HistoryOsmPrimitive p3 = getPrimitive(row);
434 if (p3 != null && p3.getTimestamp() != null)
435 return DateUtils.formatDateTime(p3.getTimestamp(), DateFormat.SHORT, DateFormat.SHORT);
436 return null;
437 case VersionTableColumnModel.COL_USER:
438 HistoryOsmPrimitive p4 = getPrimitive(row);
439 if (p4 != null) {
440 User user = p4.getUser();
441 if (user != null)
442 return user.getName();
443 }
444 return null;
445 case VersionTableColumnModel.COL_EDITOR:
446 HistoryOsmPrimitive p5 = getPrimitive(row);
447 if (p5 != null) {
448 Changeset cs = p5.getChangeset();
449 if (cs != null) {
450 return cs.get("created_by");
451 }
452 }
453 return null;
454 }
455 return null;
456 }
457
458 @Override
459 public void setValueAt(Object aValue, int row, int column) {
460 if (!((Boolean) aValue))
461 return;
462 switch (column) {
463 case 1:
464 setReferencePointInTime(row);
465 break;
466 case 2:
467 setCurrentPointInTime(row);
468 break;
469 default:
470 return;
471 }
472 fireTableDataChanged();
473 }
474
475 @Override
476 public boolean isCellEditable(int row, int column) {
477 return column >= 1 && column <= 2;
478 }
479
480 public void setReferencePointInTime(int row) {
481 if (history == null)
482 return;
483 if (row == history.getNumVersions()) {
484 if (latest != null) {
485 HistoryBrowserModel.this.setReferencePointInTime(latest);
486 }
487 return;
488 }
489 if (row < 0 || row > history.getNumVersions())
490 return;
491 HistoryOsmPrimitive reference = history.get(row);
492 HistoryBrowserModel.this.setReferencePointInTime(reference);
493 }
494
495 public void setCurrentPointInTime(int row) {
496 if (history == null)
497 return;
498 if (row == history.getNumVersions()) {
499 if (latest != null) {
500 HistoryBrowserModel.this.setCurrentPointInTime(latest);
501 }
502 return;
503 }
504 if (row < 0 || row > history.getNumVersions())
505 return;
506 HistoryOsmPrimitive current = history.get(row);
507 HistoryBrowserModel.this.setCurrentPointInTime(current);
508 }
509
510 public boolean isReferencePointInTime(int row) {
511 if (history == null)
512 return false;
513 if (row == history.getNumVersions())
514 return latest == reference;
515 if (row < 0 || row > history.getNumVersions())
516 return false;
517 HistoryOsmPrimitive p = history.get(row);
518 return p == reference;
519 }
520
521 public boolean isCurrentPointInTime(int row) {
522 if (history == null)
523 return false;
524 if (row == history.getNumVersions())
525 return latest == current;
526 if (row < 0 || row > history.getNumVersions())
527 return false;
528 HistoryOsmPrimitive p = history.get(row);
529 return p == current;
530 }
531
532 public HistoryOsmPrimitive getPrimitive(int row) {
533 if (history == null)
534 return null;
535 return isLatest(row) ? latest : history.get(row);
536 }
537
538 public boolean isLatest(int row) {
539 return row >= history.getNumVersions();
540 }
541
542 public OsmPrimitive getLatest() {
543 if (latest == null)
544 return null;
545 OsmDataLayer editLayer = Main.getLayerManager().getEditLayer();
546 if (editLayer == null)
547 return null;
548 return editLayer.data.getPrimitiveById(latest.getId(), latest.getType());
549 }
550
551 @Override
552 public int getColumnCount() {
553 return 6;
554 }
555 }
556
557 /**
558 * The table model for the tags of the version at {@link PointInTimeType#REFERENCE_POINT_IN_TIME}
559 * or {@link PointInTimeType#CURRENT_POINT_IN_TIME}
560 *
561 */
562 public class TagTableModel extends AbstractTableModel {
563
564 private List<String> keys;
565 private final PointInTimeType pointInTimeType;
566
567 protected TagTableModel(PointInTimeType type) {
568 pointInTimeType = type;
569 initKeyList();
570 }
571
572 protected void initKeyList() {
573 Set<String> keySet = new HashSet<>();
574 if (current != null) {
575 keySet.addAll(current.getTags().keySet());
576 }
577 if (reference != null) {
578 keySet.addAll(reference.getTags().keySet());
579 }
580 keys = new ArrayList<>(keySet);
581 Collections.sort(keys);
582 fireTableDataChanged();
583 }
584
585 @Override
586 public int getRowCount() {
587 if (keys == null)
588 return 0;
589 return keys.size();
590 }
591
592 @Override
593 public Object getValueAt(int row, int column) {
594 return getKeyAt(row);
595 }
596
597 /**
598 * Get the key for the given row.
599 * @param row The row
600 * @return The key in that row.
601 * @since 10637
602 */
603 public String getKeyAt(int row) {
604 return keys.get(row);
605 }
606
607 public boolean hasTag(String key) {
608 HistoryOsmPrimitive primitive = getPointInTime(pointInTimeType);
609 if (primitive == null)
610 return false;
611 return primitive.hasTag(key);
612 }
613
614 public String getValue(String key) {
615 HistoryOsmPrimitive primitive = getPointInTime(pointInTimeType);
616 if (primitive == null)
617 return null;
618 return primitive.get(key);
619 }
620
621 public boolean oppositeHasTag(String key) {
622 PointInTimeType opposite = pointInTimeType.opposite();
623 HistoryOsmPrimitive primitive = getPointInTime(opposite);
624 if (primitive == null)
625 return false;
626 return primitive.hasTag(key);
627 }
628
629 public String getOppositeValue(String key) {
630 PointInTimeType opposite = pointInTimeType.opposite();
631 HistoryOsmPrimitive primitive = getPointInTime(opposite);
632 if (primitive == null)
633 return null;
634 return primitive.get(key);
635 }
636
637 public boolean hasSameValueAsOpposite(String key) {
638 String value = getValue(key);
639 String oppositeValue = getOppositeValue(key);
640 if (value == null || oppositeValue == null)
641 return false;
642 return value.equals(oppositeValue);
643 }
644
645 public PointInTimeType getPointInTimeType() {
646 return pointInTimeType;
647 }
648
649 public boolean isCurrentPointInTime() {
650 return pointInTimeType.equals(PointInTimeType.CURRENT_POINT_IN_TIME);
651 }
652
653 public boolean isReferencePointInTime() {
654 return pointInTimeType.equals(PointInTimeType.REFERENCE_POINT_IN_TIME);
655 }
656
657 @Override
658 public int getColumnCount() {
659 return 2;
660 }
661 }
662
663 protected void setLatest(HistoryOsmPrimitive latest) {
664 if (latest == null) {
665 if (this.current == this.latest) {
666 this.current = history != null ? history.getLatest() : null;
667 }
668 if (this.reference == this.latest) {
669 this.reference = history != null ? history.getLatest() : null;
670 }
671 this.latest = null;
672 } else {
673 if (this.current == this.latest) {
674 this.current = latest;
675 }
676 if (this.reference == this.latest) {
677 this.reference = latest;
678 }
679 this.latest = latest;
680 }
681 fireModelChange();
682 }
683
684 /**
685 * Removes this model as listener for data change and layer change events.
686 *
687 */
688 public void unlinkAsListener() {
689 OsmDataLayer editLayer = Main.getLayerManager().getEditLayer();
690 if (editLayer != null) {
691 editLayer.data.removeDataSetListener(this);
692 }
693 Main.getLayerManager().removeActiveLayerChangeListener(this);
694 }
695
696 /* ---------------------------------------------------------------------- */
697 /* DataSetListener */
698 /* ---------------------------------------------------------------------- */
699 @Override
700 public void nodeMoved(NodeMovedEvent event) {
701 Node node = event.getNode();
702 if (!node.isNew() && node.getId() == history.getId()) {
703 setLatest(new HistoryPrimitiveBuilder().build(node));
704 }
705 }
706
707 @Override
708 public void primitivesAdded(PrimitivesAddedEvent event) {
709 for (OsmPrimitive p: event.getPrimitives()) {
710 if (canShowAsLatest(p)) {
711 setLatest(new HistoryPrimitiveBuilder().build(p));
712 }
713 }
714 }
715
716 @Override
717 public void primitivesRemoved(PrimitivesRemovedEvent event) {
718 for (OsmPrimitive p: event.getPrimitives()) {
719 if (!p.isNew() && p.getId() == history.getId()) {
720 setLatest(null);
721 }
722 }
723 }
724
725 @Override
726 public void relationMembersChanged(RelationMembersChangedEvent event) {
727 Relation r = event.getRelation();
728 if (!r.isNew() && r.getId() == history.getId()) {
729 setLatest(new HistoryPrimitiveBuilder().build(r));
730 }
731 }
732
733 @Override
734 public void tagsChanged(TagsChangedEvent event) {
735 OsmPrimitive prim = event.getPrimitive();
736 if (!prim.isNew() && prim.getId() == history.getId()) {
737 setLatest(new HistoryPrimitiveBuilder().build(prim));
738 }
739 }
740
741 @Override
742 public void wayNodesChanged(WayNodesChangedEvent event) {
743 Way way = event.getChangedWay();
744 if (!way.isNew() && way.getId() == history.getId()) {
745 setLatest(new HistoryPrimitiveBuilder().build(way));
746 }
747 }
748
749 @Override
750 public void dataChanged(DataChangedEvent event) {
751 if (history == null)
752 return;
753 OsmPrimitive primitive = event.getDataset().getPrimitiveById(history.getId(), history.getType());
754 HistoryOsmPrimitive latest;
755 if (canShowAsLatest(primitive)) {
756 latest = new HistoryPrimitiveBuilder().build(primitive);
757 } else {
758 latest = null;
759 }
760 setLatest(latest);
761 fireModelChange();
762 }
763
764 @Override
765 public void otherDatasetChange(AbstractDatasetChangedEvent event) {
766 // Irrelevant
767 }
768
769 /* ---------------------------------------------------------------------- */
770 /* ActiveLayerChangeListener */
771 /* ---------------------------------------------------------------------- */
772 @Override
773 public void activeOrEditLayerChanged(ActiveLayerChangeEvent e) {
774 Layer oldLayer = e.getPreviousActiveLayer();
775 if (oldLayer instanceof OsmDataLayer) {
776 OsmDataLayer l = (OsmDataLayer) oldLayer;
777 l.data.removeDataSetListener(this);
778 }
779 Layer newLayer = e.getSource().getActiveLayer();
780 if (!(newLayer instanceof OsmDataLayer)) {
781 latest = null;
782 fireModelChange();
783 return;
784 }
785 OsmDataLayer l = (OsmDataLayer) newLayer;
786 l.data.addDataSetListener(this);
787 OsmPrimitive primitive = history != null ? l.data.getPrimitiveById(history.getId(), history.getType()) : null;
788 HistoryOsmPrimitive newLatest;
789 if (canShowAsLatest(primitive)) {
790 newLatest = new HistoryPrimitiveBuilder().build(primitive);
791 } else {
792 newLatest = null;
793 }
794 setLatest(newLatest);
795 fireModelChange();
796 }
797
798 /**
799 * Creates a {@link HistoryOsmPrimitive} from a {@link OsmPrimitive}
800 *
801 */
802 static class HistoryPrimitiveBuilder extends AbstractVisitor {
803 private HistoryOsmPrimitive clone;
804
805 @Override
806 public void visit(Node n) {
807 clone = new HistoryNode(n.getId(), n.getVersion(), n.isVisible(), getCurrentUser(), 0, null, n.getCoor(), false);
808 clone.setTags(n.getKeys());
809 }
810
811 @Override
812 public void visit(Relation r) {
813 clone = new HistoryRelation(r.getId(), r.getVersion(), r.isVisible(), getCurrentUser(), 0, null, false);
814 clone.setTags(r.getKeys());
815 HistoryRelation hr = (HistoryRelation) clone;
816 for (RelationMember rm : r.getMembers()) {
817 hr.addMember(new RelationMemberData(rm.getRole(), rm.getType(), rm.getUniqueId()));
818 }
819 }
820
821 @Override
822 public void visit(Way w) {
823 clone = new HistoryWay(w.getId(), w.getVersion(), w.isVisible(), getCurrentUser(), 0, null, false);
824 clone.setTags(w.getKeys());
825 for (Node n: w.getNodes()) {
826 ((HistoryWay) clone).addNode(n.getUniqueId());
827 }
828 }
829
830 private static User getCurrentUser() {
831 UserInfo info = JosmUserIdentityManager.getInstance().getUserInfo();
832 return info == null ? User.getAnonymous() : User.createOsmUser(info.getId(), info.getDisplayName());
833 }
834
835 public HistoryOsmPrimitive build(OsmPrimitive primitive) {
836 primitive.accept(this);
837 return clone;
838 }
839 }
840
841}
Note: See TracBrowser for help on using the repository browser.