ModularMovementModelViz.java 10.6 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
/*
 * Copyright (c) 2005-2015 KOM – Multimedia Communications Lab
 *
 * This file is part of PeerfactSim.KOM.
 * 
 * PeerfactSim.KOM is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * any later version.
 * 
 * PeerfactSim.KOM is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 * 
 * You should have received a copy of the GNU General Public License
 * along with PeerfactSim.KOM.  If not, see <http://www.gnu.org/licenses/>.
 *
 */

package de.tud.kom.p2psim.impl.topology.movement.modularosm;

23
import java.awt.AlphaComposite;
24
import java.awt.BasicStroke;
25
26
27
28
29
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Point;
import java.awt.RenderingHints;
30
import java.awt.geom.Point2D;
31
32
import java.util.LinkedHashMap;
import java.util.Map;
33
import javax.swing.JCheckBoxMenuItem;
34
import javax.swing.JComponent;
35
import javax.swing.JMenu;
36
37
38
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import de.tud.kom.p2psim.api.topology.movement.SimLocationActuator;
39
import de.tud.kom.p2psim.impl.topology.movement.local.AbstractLocalMovementStrategy;
40
import de.tud.kom.p2psim.impl.topology.util.PositionVector;
41
import de.tud.kom.p2psim.impl.topology.views.VisualizationTopologyView;
42
import de.tud.kom.p2psim.impl.topology.views.VisualizationTopologyView.VisualizationInjector;
43
import de.tud.kom.p2psim.impl.topology.views.visualization.ui.VisualizationComponent;
44
45
46
47
48
import de.tud.kom.p2psim.impl.topology.views.visualization.world.NodeVisInteractionListener;
import de.tudarmstadt.maki.simonstrator.api.Host;
import de.tudarmstadt.maki.simonstrator.api.Oracle;
import de.tudarmstadt.maki.simonstrator.api.common.graph.INodeID;
import de.tudarmstadt.maki.simonstrator.api.component.ComponentNotAvailableException;
49
import de.tudarmstadt.maki.simonstrator.api.component.sensor.location.AttractionPoint;
50
51
52
53
import de.tudarmstadt.maki.simonstrator.api.component.sensor.location.Location;
import de.tudarmstadt.maki.simonstrator.api.component.sensor.route.Route;
import de.tudarmstadt.maki.simonstrator.api.component.sensor.route.Route.RouteSegment;
import de.tudarmstadt.maki.simonstrator.api.component.sensor.route.RouteSensor;
54
55

/**
56
57
 * Visualization Component of the Attraction Points in the Modular Movement
 * Model.
58
59
 * 
 * 
60
 * @author Martin Hellwig, Bjoern Richerzhagen
61
62
 * @version 1.0, 02.07.2015
 */
63
public class ModularMovementModelViz extends JComponent
64
		implements VisualizationComponent, NodeVisInteractionListener {
65

Clemens Krug's avatar
Clemens Krug committed
66
	protected ModularMovementModel movementModel;
67

68
69
70
71
	protected boolean showAttractionPoints = true;

	protected boolean showNodePositions = true;

72
	protected boolean showTrajectories = false;
73
74
	
	protected boolean setupCalculatesTrajectories = false;
75

76
77
	private JMenu menu;

78
79
	private Map<Long, TrajectoryVis> trajectoryVisualizations = new LinkedHashMap<>();

80
81
	private final static int NODE_PAD = 2;

82
	private final static int ATTR_PAD = 5;
83
84

	private static Color COLOR_ATTR_POINT = Color.decode("#4A7B9D");
85

86
	public ModularMovementModelViz() {
Clemens Krug's avatar
Clemens Krug committed
87
88
89
        init();
	}

90
	protected void init() {
Clemens Krug's avatar
Clemens Krug committed
91
92
        setBounds(0, 0, VisualizationInjector.getWorldX(),
                VisualizationInjector.getWorldY());
93
		VisualizationInjector.addInteractionListener(this);
Clemens Krug's avatar
Clemens Krug committed
94
95
96
97
98
99
        setOpaque(true);
        setVisible(true);

        menu = new JMenu("Mobility Model");
        menu.add(createCheckboxAp());
        menu.add(createCheckboxNodePositions());
100
		menu.add(createCheckboxTrajectories());
Clemens Krug's avatar
Clemens Krug committed
101
102
    }

103
	public ModularMovementModelViz(ModularMovementModel model) {
Clemens Krug's avatar
Clemens Krug committed
104
		init();
105
		this.movementModel = model;
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
	}

	private JCheckBoxMenuItem createCheckboxAp() {
		final JCheckBoxMenuItem checkBox = new JCheckBoxMenuItem(
				"show attraction points", showAttractionPoints);
		checkBox.addChangeListener(new ChangeListener() {
			@Override
			public void stateChanged(ChangeEvent e) {
				showAttractionPoints = checkBox.isSelected();
				VisualizationInjector.invalidate();
			}
		});
		return checkBox;
	}

	private JCheckBoxMenuItem createCheckboxNodePositions() {
		final JCheckBoxMenuItem checkBox = new JCheckBoxMenuItem(
				"show node positions", showNodePositions);
		checkBox.addChangeListener(new ChangeListener() {
			@Override
			public void stateChanged(ChangeEvent e) {
				showNodePositions = checkBox.isSelected();
				VisualizationInjector.invalidate();
			}
		});
		return checkBox;
132
133
	}

134
	private JCheckBoxMenuItem createCheckboxTrajectories() {
135
		setupCalculatesTrajectories = AbstractLocalMovementStrategy.isCalculatingRouteSegments();
136
137
138
139
140
141
		final JCheckBoxMenuItem checkBox = new JCheckBoxMenuItem(
				"show node trajectories", showTrajectories);
		checkBox.addChangeListener(new ChangeListener() {
			@Override
			public void stateChanged(ChangeEvent e) {
				showTrajectories = checkBox.isSelected();
142
143
144
145
				if (!setupCalculatesTrajectories) {
					AbstractLocalMovementStrategy
							.setCalculationOfRouteSegments(showTrajectories);
				}
146
147
148
				VisualizationInjector.invalidate();
			}
		});
149
150
151
152
		if (!setupCalculatesTrajectories) {
			AbstractLocalMovementStrategy
					.setCalculationOfRouteSegments(showTrajectories);
		}
153
154
155
		return checkBox;
	}

156
157
158
159
160
161
162
	@Override
	public void paint(Graphics g) {
		super.paintComponent(g);
		Graphics2D g2 = (Graphics2D) g;
		g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING,
				RenderingHints.VALUE_ANTIALIAS_ON);

163
		if (showAttractionPoints) {
Clemens Krug's avatar
Clemens Krug committed
164
			drawAttractionPoints(g2);
165
		}
166

167
		if (showNodePositions) {
Clemens Krug's avatar
Clemens Krug committed
168
169
170
171
            for (SimLocationActuator comp : movementModel
                    .getAllLocationActuators()) {
                drawNodePosition(g2, comp);
            }
172
		}
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206

		if (showTrajectories) {
			for (TrajectoryVis tVis : trajectoryVisualizations.values()) {
				tVis.drawTrajectory(g2);
			}
		}
	}

	@Override
	public void onHostClick(long hostID, boolean isActive) {
		if (isActive) {
			Host h = Oracle.getHostByID(INodeID.get(hostID));
			try {
				RouteSensor routeSensor = h.getComponent(RouteSensor.class);
				trajectoryVisualizations.put(hostID,
						new TrajectoryVis(routeSensor));
			} catch (ComponentNotAvailableException e) {
				// ignore.
			}

		} else {
			trajectoryVisualizations.remove(hostID);
		}
	}

	/**
	 * Visualizes the movement trajectory of a given node.
	 * 
	 * @author Bjoern Richerzhagen
	 * @version 1.0, Aug 10, 2017
	 */
	private class TrajectoryVis {

		private final RouteSensor routeSensor;
207
		
208
209
210
211
212
213
		public TrajectoryVis(RouteSensor routeSensor) {
			this.routeSensor = routeSensor;
		}

		public void drawTrajectory(Graphics2D g2) {
			Route rt = routeSensor.getRoute();
Björn Richerzhagen's avatar
Björn Richerzhagen committed
214
215
216
			if (!rt.hasSegmentInformation()) {
				return;
			}
217
			boolean alt = false;
218
219
			for (RouteSegment segment : rt.getSegmentsAhead()) {
				Location lastLoc = null;
220
221
222
223
224
				g2.setColor(Color.MAGENTA);
				if (alt) {
					g2.setColor(Color.ORANGE);
				}
				alt = !alt;
225
226
227
				for (Location loc : segment.getSegmentLocations()) {
					if (lastLoc == null) {
						lastLoc = loc;
228
229
						g2.drawString(segment.getSegmentId(),
								VisualizationInjector
230
										.scaleValue(lastLoc.getLongitudeOrX()),
231
								VisualizationInjector
232
										.scaleValue(lastLoc.getLatitudeOrY()));
233
234
						continue;
					}
235
					g2.setStroke(new BasicStroke(3.0f));
236
237
					g2.drawLine(
							VisualizationInjector
238
									.scaleValue(lastLoc.getLongitudeOrX()),
239
							VisualizationInjector
240
									.scaleValue(lastLoc.getLatitudeOrY()),
241
							VisualizationInjector
242
									.scaleValue(loc.getLongitudeOrX()),
243
							VisualizationInjector
244
									.scaleValue(loc.getLatitudeOrY()));
245
246
247
248
249
					lastLoc = loc;
				}
			}
		}

250
	}
251

Clemens Krug's avatar
Clemens Krug committed
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
    /**
     * Provides access super.paintComponent() for extending classes.
     * @param g the Graphics object for painting.
     */
    protected void paintSuper(Graphics g)
    {
        super.paintComponent(g);
    }

    /**
     * Draws the attraction points. This method has been extracted from paint()
     * to make it possible for extending class to override only this bit while leaving everything
     * else untouched.
     * @param g2 the Graphics2D object for painting.
     */
	protected void drawAttractionPoints(Graphics2D g2)
    {
        for (AttractionPoint aPoint : movementModel.getAttractionPoints()) {
            Point point = ((PositionVector) aPoint).asPoint();
            // draw border
            g2.setColor(Color.BLACK);
            g2.setFont(VisualizationTopologyView.FONT_MEDIUM);
            g2.drawString(aPoint.getName(),
                    VisualizationInjector.scaleValue(point.x) - ATTR_PAD,
                    VisualizationInjector.scaleValue(point.y) - ATTR_PAD);
            g2.setColor(COLOR_ATTR_POINT);
278
279
            float alpha = 0.25f + (float) (aPoint.getWeight() / 2);
            g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha));
Clemens Krug's avatar
Clemens Krug committed
280
281
282
283
            g2.fillOval(
                    VisualizationInjector.scaleValue(point.x) - ATTR_PAD,
                    VisualizationInjector.scaleValue(point.y) - ATTR_PAD,
                    ATTR_PAD * 2 + 1, ATTR_PAD * 2 + 1);
284
            g2.setColor(COLOR_ATTR_POINT);
285
            int radius = VisualizationInjector.scaleValue(aPoint.getRadius()) + ATTR_PAD;
286
287
288
            g2.drawOval(VisualizationInjector.scaleValue(point.x) - radius,
                    VisualizationInjector.scaleValue(point.y) - radius,
                    radius * 2 + 1, radius * 2 + 1);
Clemens Krug's avatar
Clemens Krug committed
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303

        }
    }

    /**
     * Draws the position of the node. This method has been extracted from paint()
     * to make it possible for extending class to override only this bit while leaving everything
     * else untouched. Instead of painting all of the nodes, only a specific one gets painted. This allows
     * extending class to fall back to this implementation if one specific node could not be drawn for whatever reason.
     * @param g2 the Graphics2D object for painting.
     */
	protected void drawNodePosition(Graphics2D g2, SimLocationActuator comp)
	{
        Point2D pt = comp.getRealPosition().asPoint();
        g2.setColor(Color.GRAY);
304
305
        g2.fillOval(VisualizationInjector.scaleValue(pt.getX()) - NODE_PAD,
                VisualizationInjector.scaleValue(pt.getY()) - NODE_PAD, NODE_PAD * 2 + 1,
Clemens Krug's avatar
Clemens Krug committed
306
307
308
309
310
311
312
313
                NODE_PAD * 2 + 1);
	}

	protected void setMovementModel(ModularMovementModel model)
    {
        movementModel = model;
    }

314
315
316
317
318
319
320
321
322
323
324
325
	@Override
	public String getDisplayName() {
		return "Mobility Model";
	}

	@Override
	public JComponent getComponent() {
		return this;
	}

	@Override
	public JMenu getCustomMenu() {
326
		return menu;
327
328
329
330
331
332
	}

	@Override
	public boolean isHidden() {
		return false;
	}
333
}