ModularMovementModelViz.java 9.93 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
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229

		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();
			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;
				}
			}
		}

230
	}
231

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

        }
    }

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

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

294
295
296
297
298
299
300
301
302
303
304
305
	@Override
	public String getDisplayName() {
		return "Mobility Model";
	}

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

	@Override
	public JMenu getCustomMenu() {
306
		return menu;
307
308
309
310
311
312
	}

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