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.Stroke;
31
import java.awt.geom.Point2D;
32
33
import java.util.LinkedHashMap;
import java.util.Map;
34

35
import javax.swing.JCheckBoxMenuItem;
36
import javax.swing.JComponent;
37
import javax.swing.JMenu;
38
39
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
40

41
import de.tud.kom.p2psim.api.topology.movement.SimLocationActuator;
42
import de.tud.kom.p2psim.impl.topology.PositionVector;
43
import de.tud.kom.p2psim.impl.topology.movement.local.AbstractLocalMovementStrategy;
44
import de.tud.kom.p2psim.impl.topology.views.VisualizationTopologyView;
45
import de.tud.kom.p2psim.impl.topology.views.VisualizationTopologyView.VisualizationInjector;
46
import de.tud.kom.p2psim.impl.topology.views.visualization.ui.VisualizationComponent;
47
48
49
50
51
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;
52
import de.tudarmstadt.maki.simonstrator.api.component.sensor.location.AttractionPoint;
53
54
55
56
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;
57
58

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

Clemens Krug's avatar
Clemens Krug committed
69
	protected ModularMovementModel movementModel;
70

71
72
73
74
	protected boolean showAttractionPoints = true;

	protected boolean showNodePositions = true;

75
	protected boolean showTrajectories = false;
76
77
	
	protected boolean setupCalculatesTrajectories = false;
78

79
80
	private JMenu menu;

81
82
	private Map<Long, TrajectoryVis> trajectoryVisualizations = new LinkedHashMap<>();

83
84
	private final static int NODE_PAD = 2;

85
	private final static int ATTR_PAD = 5;
86
87

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

89
	public ModularMovementModelViz() {
Clemens Krug's avatar
Clemens Krug committed
90
91
92
        init();
	}

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

        menu = new JMenu("Mobility Model");
        menu.add(createCheckboxAp());
        menu.add(createCheckboxNodePositions());
103
		menu.add(createCheckboxTrajectories());
Clemens Krug's avatar
Clemens Krug committed
104
105
    }

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

	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;
135
136
	}

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

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

166
		if (showAttractionPoints) {
Clemens Krug's avatar
Clemens Krug committed
167
			drawAttractionPoints(g2);
168
		}
169

170
		if (showNodePositions) {
Clemens Krug's avatar
Clemens Krug committed
171
172
173
174
            for (SimLocationActuator comp : movementModel
                    .getAllLocationActuators()) {
                drawNodePosition(g2, comp);
            }
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
207
208
209

		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;
210
		
211
212
213
214
215
216
		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
217
218
219
			if (!rt.hasSegmentInformation()) {
				return;
			}
220
			boolean alt = false;
221
222
			for (RouteSegment segment : rt.getSegmentsAhead()) {
				Location lastLoc = null;
223
224
225
226
227
				g2.setColor(Color.MAGENTA);
				if (alt) {
					g2.setColor(Color.ORANGE);
				}
				alt = !alt;
228
229
230
				for (Location loc : segment.getSegmentLocations()) {
					if (lastLoc == null) {
						lastLoc = loc;
231
232
233
234
235
						g2.drawString(segment.getSegmentId(),
								VisualizationInjector
										.scaleValue(lastLoc.getLongitude()),
								VisualizationInjector
										.scaleValue(lastLoc.getLatitude()));
236
237
						continue;
					}
238
					g2.setStroke(new BasicStroke(3.0f));
239
240
241
242
243
244
245
246
247
248
249
250
251
252
					g2.drawLine(
							VisualizationInjector
									.scaleValue(lastLoc.getLongitude()),
							VisualizationInjector
									.scaleValue(lastLoc.getLatitude()),
							VisualizationInjector
									.scaleValue(loc.getLongitude()),
							VisualizationInjector
									.scaleValue(loc.getLatitude()));
					lastLoc = loc;
				}
			}
		}

253
	}
254

Clemens Krug's avatar
Clemens Krug committed
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
    /**
     * 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);
281
282
            float alpha = 0.25f + (float) (aPoint.getWeight() / 2);
            g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha));
Clemens Krug's avatar
Clemens Krug committed
283
284
285
286
            g2.fillOval(
                    VisualizationInjector.scaleValue(point.x) - ATTR_PAD,
                    VisualizationInjector.scaleValue(point.y) - ATTR_PAD,
                    ATTR_PAD * 2 + 1, ATTR_PAD * 2 + 1);
287
            g2.setColor(COLOR_ATTR_POINT);
288
            int radius = VisualizationInjector.scaleValue(aPoint.getRadius()) + ATTR_PAD;
289
290
291
            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
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306

        }
    }

    /**
     * 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);
307
308
        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
309
310
311
312
313
314
315
316
                NODE_PAD * 2 + 1);
	}

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

317
318
319
320
321
322
323
324
325
326
327
328
	@Override
	public String getDisplayName() {
		return "Mobility Model";
	}

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

	@Override
	public JMenu getCustomMenu() {
329
		return menu;
330
331
332
333
334
335
	}

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