ModularMovementModelViz.java 10.8 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
import java.awt.Color;
26
import java.awt.Composite;
27
28
29
30
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Point;
import java.awt.RenderingHints;
31
import java.awt.geom.Point2D;
32
33
import java.util.LinkedHashMap;
import java.util.Map;
34
import javax.swing.JCheckBoxMenuItem;
35
import javax.swing.JComponent;
36
import javax.swing.JMenu;
37
38
39
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import de.tud.kom.p2psim.api.topology.movement.SimLocationActuator;
40
import de.tud.kom.p2psim.impl.topology.movement.local.AbstractLocalMovementStrategy;
41
import de.tud.kom.p2psim.impl.topology.util.PositionVector;
42
import de.tud.kom.p2psim.impl.topology.views.VisualizationTopologyView;
43
import de.tud.kom.p2psim.impl.topology.views.VisualizationTopologyView.VisualizationInjector;
44
import de.tud.kom.p2psim.impl.topology.views.visualization.ui.VisualizationComponent;
45
46
47
48
49
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;
50
import de.tudarmstadt.maki.simonstrator.api.component.sensor.location.AttractionPoint;
51
52
53
54
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;
55
56

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

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

69
70
71
72
	protected boolean showAttractionPoints = true;

	protected boolean showNodePositions = true;

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

77
78
	private JMenu menu;

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

81
82
	private final static int NODE_PAD = 2;

83
	private final static int ATTR_PAD = 5;
84
85

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

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

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

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

104
	public ModularMovementModelViz(ModularMovementModel model) {
Clemens Krug's avatar
Clemens Krug committed
105
		init();
106
		this.movementModel = model;
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
132
	}

	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;
133
134
	}

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

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

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

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

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

251
	}
252

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

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

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

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

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

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

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