ModularMovementModelViz.java 9.99 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
25
26
27
28
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Point;
import java.awt.RenderingHints;
29
import java.awt.geom.Point2D;
30
31
import java.util.LinkedHashMap;
import java.util.Map;
32

33
import javax.swing.JCheckBoxMenuItem;
34
import javax.swing.JComponent;
35
import javax.swing.JMenu;
36
37
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
38

39
import de.tud.kom.p2psim.api.topology.movement.SimLocationActuator;
40
import de.tud.kom.p2psim.impl.topology.PositionVector;
41
import de.tud.kom.p2psim.impl.topology.movement.local.AbstractLocalMovementStrategy;
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
74
	protected boolean showTrajectories = false;

75
76
	private JMenu menu;

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

79
80
	private final static int NODE_PAD = 2;

81
	private final static int ATTR_PAD = 5;
82
83

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

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

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

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

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

	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;
131
132
	}

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

150
151
152
153
154
155
156
	@Override
	public void paint(Graphics g) {
		super.paintComponent(g);
		Graphics2D g2 = (Graphics2D) g;
		g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING,
				RenderingHints.VALUE_ANTIALIAS_ON);

157
		if (showAttractionPoints) {
Clemens Krug's avatar
Clemens Krug committed
158
			drawAttractionPoints(g2);
159
		}
160

161
		if (showNodePositions) {
Clemens Krug's avatar
Clemens Krug committed
162
163
164
165
            for (SimLocationActuator comp : movementModel
                    .getAllLocationActuators()) {
                drawNodePosition(g2, comp);
            }
166
		}
167
168
169
170
171
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
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;

		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
208
209
210
			if (!rt.hasSegmentInformation()) {
				return;
			}
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
			for (RouteSegment segment : rt.getSegmentsAhead()) {
				Location lastLoc = null;
				for (Location loc : segment.getSegmentLocations()) {
					if (lastLoc == null) {
						lastLoc = loc;
						continue;
					}
					g2.setColor(Color.MAGENTA);
					g2.drawLine(
							VisualizationInjector
									.scaleValue(lastLoc.getLongitude()),
							VisualizationInjector
									.scaleValue(lastLoc.getLatitude()),
							VisualizationInjector
									.scaleValue(loc.getLongitude()),
							VisualizationInjector
									.scaleValue(loc.getLatitude()));
					lastLoc = loc;
				}
			}
		}

233
	}
234

Clemens Krug's avatar
Clemens Krug committed
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
    /**
     * 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);
261
262
            float alpha = 0.25f + (float) (aPoint.getWeight() / 2);
            g2.setComposite(AlphaComposite.getInstance(AlphaComposite.SRC_OVER, alpha));
Clemens Krug's avatar
Clemens Krug committed
263
264
265
266
            g2.fillOval(
                    VisualizationInjector.scaleValue(point.x) - ATTR_PAD,
                    VisualizationInjector.scaleValue(point.y) - ATTR_PAD,
                    ATTR_PAD * 2 + 1, ATTR_PAD * 2 + 1);
267
            g2.setColor(COLOR_ATTR_POINT);
268
            int radius = VisualizationInjector.scaleValue(aPoint.getRadius()) + ATTR_PAD;
269
270
271
            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
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286

        }
    }

    /**
     * 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);
287
288
        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
289
290
291
292
293
294
295
296
                NODE_PAD * 2 + 1);
	}

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

297
298
299
300
301
302
303
304
305
306
307
308
	@Override
	public String getDisplayName() {
		return "Mobility Model";
	}

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

	@Override
	public JMenu getCustomMenu() {
309
		return menu;
310
311
312
313
314
315
	}

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