Project

General

Profile

Statistics
| Revision:

root / trunk / code / projects / colonet / client / Colonet.java @ 938

History | View | Annotate | Download (38.7 KB)

1
import javax.swing.*;
2
import javax.swing.event.*;
3
import javax.imageio.*;
4
import java.awt.*;
5
import java.awt.image.*;
6
import java.awt.event.*;
7
import java.net.*;
8
import java.io.*;
9
import java.util.*;
10

    
11
/**
12
* The Colonet Graphical User Interface Applet for use locally and over an internet connection.
13
* @author Gregory Tress
14
*
15
* To generate javadoc on this file or other java files, use javadoc *.java -d doc, where doc
16
* is the name of the folder into which the files should be written.
17
*/
18
public class Colonet extends JApplet implements ActionListener, MouseInputListener, KeyListener {
19

    
20
        // Connection
21
        JTextField txtHost;
22
        JTextField txtPort;
23
        JButton btnConnect;
24
        JButton btnGetXBeeIDs;
25
        JLabel lblConnectionStatus;
26
        JTextArea txtInfo;
27
        JPanel panelConnect;
28
        JPanel panelServerInterface;
29
        Socket socket;
30
        DataUpdater dataUpdater;
31

    
32
        // Control
33
        JTabbedPane tabPaneControl;
34
        JPanel panelRobotControl;
35
        JPanel panelRobotDirection;
36
        JPanel panelRobotDirectionButtons;
37
        JPanel panelRobotCommands;
38
        JButton btnF, btnB, btnL, btnR, btnActivate;
39
        JLabel lblBattery;
40
        JLabel lblSelected;
41
        BatteryIcon batteryIcon;
42
        JPanel panelBattery;
43
        VectorController vectorController;
44
        BufferedImage imageVectorControl;
45
        JButton btnAssignID;
46
        JButton btnLocateStation;
47
        boolean setStation;
48
        ChargingStation station;
49
        boolean setWaypoint;
50
        int setWaypointID;
51
        JButton btnSetBounds;
52
        JButton btnClearBounds;
53
        RobotBoundary boundary;
54
        JButton btnCommand_MoveTo;
55
        JButton btnCommand_MoveAll;
56
        JButton btnCommand_StopTask;
57
        JButton btnCommand_ResumeTask;
58
        JButton btnCommand_ChargeNow;
59
        JButton btnCommand_StopCharging;
60

    
61
        // Task Manager
62
        JPanel panelTaskManager;
63
        JScrollPane spTaskManager;
64
        JPanel panelTaskManagerControls;
65
        JPanel panelTaskManagerControlsPriority;
66
        DefaultListModel taskListModel;
67
        JList taskList;
68
        JButton btnAddTask;
69
        JButton btnRemoveTask;
70
        JButton btnMoveTaskUp;
71
        JButton btnMoveTaskDown;
72
        JButton btnUpdateTasks;
73
        TaskAddWindow taskAddWindow;
74

    
75
        // Webcam
76
        WebcamPanel panelWebcam;
77
        GraphicsConfiguration gc;
78
        JTabbedPane tabPaneMain;
79
        WebcamLoader webcamLoader;
80
        
81
        // Robots
82
        volatile int selectedBot;         //the user has selected this bot graphically
83
        volatile RobotList robotIcons;         //contains boundary shapes around bots for click detection
84
        volatile int[] xbeeID;
85

    
86
        Colonet self = this;
87
        volatile ColonetServerInterface csi;
88

    
89
        public void init () {
90
                // Set the default look and feel
91
                String laf = ColonetConstants.LOOK_AND_FEEL;
92
                try {
93
                        UIManager.setLookAndFeel(laf);
94
                } catch (UnsupportedLookAndFeelException exc) {
95
                        System.err.println ("Warning: UnsupportedLookAndFeel: " + laf);
96
                } catch (Exception exc) {
97
                        System.err.println ("Error loading " + laf + ": " + exc);
98
                }
99

    
100
                // We should invoke and wait to avoid browser display difficulties
101
                Runnable r = new Runnable() {
102
                        public void run() {
103
                                createAndShowGUI();
104
                        }
105
                };
106

    
107
                try {
108
                        SwingUtilities.invokeAndWait(r);
109
                } catch (InterruptedException e) {
110
                        //Not really sure why we would be in this situation
111
                        System.out.println("InterruptedException in init: " + e);
112
                } catch (java.lang.reflect.InvocationTargetException e) {
113
                        //This could happen for various reasons if there is a problem in createAndShowGUI
114
                        e.printStackTrace();
115
                }
116
        }
117

    
118
        public void destroy () {
119
                if (csi != null) {
120
                        csi.disconnect();
121
                }
122
        }
123

    
124
        private synchronized void createAndShowGUI () {
125
                // Webcam
126
                gc = GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDefaultConfiguration();
127
                panelWebcam = new WebcamPanel();
128
                tabPaneMain = new JTabbedPane();
129
                tabPaneMain.setFont(new Font("arial", Font.PLAIN, 14));
130
                tabPaneMain.add(panelWebcam, "Webcam");
131

    
132
                // Robots
133
                selectedBot = -1;
134
                robotIcons = new RobotList();
135

    
136
                // Connection area
137
                txtInfo = new JTextArea();
138
                txtInfo.setBorder(BorderFactory.createTitledBorder("Info"));
139
                txtHost = new JTextField(this.getDocumentBase().getHost());
140
                txtHost.setBorder(BorderFactory.createTitledBorder("Host"));
141
                txtPort = new JTextField("10123");
142
                txtPort.setBorder(BorderFactory.createTitledBorder("Port"));
143
                btnConnect = new JButton("Connect");
144
                btnConnect.setFont(new Font("arial", Font.BOLD, 16));
145
                btnGetXBeeIDs = new JButton("Get XBee IDs");
146
                getRootPane().setDefaultButton(btnConnect);
147
                lblConnectionStatus = new JLabel("Status: Offline");
148
                panelConnect = new JPanel();
149
                panelConnect.setLayout(new GridLayout(6,1));
150
                panelConnect.add(lblConnectionStatus);
151
                panelConnect.add(txtHost);
152
                panelConnect.add(txtPort);
153
                panelConnect.add(btnConnect);
154
                panelServerInterface = new JPanel();
155
                panelServerInterface.setLayout(new GridLayout(2,1));
156
                panelServerInterface.add(panelConnect);
157
                panelServerInterface.add(txtInfo);
158

    
159
                // Robot direction panel
160
                panelRobotDirection = new JPanel();
161
                panelRobotDirectionButtons = new JPanel();
162

    
163
                Font f = new Font(null, Font.PLAIN, 16);
164
                btnF = new JButton("\u2191");
165
                btnF.setFont(f);
166
                btnB = new JButton("\u2193");
167
                btnB.setFont(f);
168
                btnL = new JButton("\u2190");
169
                btnL.setFont(f);
170
                btnR = new JButton("\u2192");
171
                btnR.setFont(f);
172
                btnActivate = new JButton("\u25A0");
173
                btnActivate.setFont(new Font(null, Font.BOLD, 24));
174

    
175
                panelRobotDirectionButtons.setLayout(new GridLayout(1,5));
176
                panelRobotDirectionButtons.add(btnActivate);
177
                panelRobotDirectionButtons.add(btnF);
178
                panelRobotDirectionButtons.add(btnB);
179
                panelRobotDirectionButtons.add(btnL);
180
                panelRobotDirectionButtons.add(btnR);
181

    
182
                imageVectorControl = gc.createCompatibleImage(ColonetConstants.VECTOR_CONTROLLER_WIDTH, ColonetConstants.VECTOR_CONTROLLER_HEIGHT);
183
                vectorController = new VectorController(imageVectorControl, self);
184
                panelRobotDirection.setLayout(new BorderLayout());
185
                panelRobotDirection.add(vectorController, BorderLayout.CENTER);
186
                panelRobotDirection.add(panelRobotDirectionButtons, BorderLayout.SOUTH);
187

    
188
                // Robot Control and Commands
189
                panelRobotCommands = new JPanel();
190
                panelRobotCommands.setLayout(new GridLayout(6,2));
191
                // Battery subset
192
                batteryIcon = new BatteryIcon(0);
193
                lblBattery = new JLabel(batteryIcon);
194
                lblSelected = new JLabel("None");
195
                // Management subset
196
                setStation = false;
197
                setWaypoint = false;
198
                setWaypointID = -1;
199
                btnAssignID = new JButton("Assign ID");
200
                btnLocateStation = new JButton("Identify Station");
201
                boundary = new RobotBoundary();
202
                btnSetBounds = new JButton("Set Boundary");
203
                btnClearBounds = new JButton("Clear Boundary");
204
                // Control subset
205
                btnCommand_MoveTo = new JButton("Move to ...");
206
                btnCommand_MoveAll = new JButton("Move all ...");
207
                btnCommand_StopTask = new JButton("Stop Current Task");
208
                btnCommand_ResumeTask = new JButton("Resume Current Task");
209
                btnCommand_ChargeNow = new JButton("Recharge Now");
210
                btnCommand_StopCharging = new JButton("Stop Recharging");
211
                panelRobotCommands.add(new JLabel("Battery Level: "));
212
                panelRobotCommands.add(lblBattery);
213
                panelRobotCommands.add(new JLabel("Selected Icon: "));
214
                panelRobotCommands.add(lblSelected);
215
                panelRobotCommands.add(btnAssignID);
216
    panelRobotCommands.add(btnLocateStation);
217
                panelRobotCommands.add(btnCommand_ChargeNow);
218
                panelRobotCommands.add(btnCommand_StopCharging);
219
                panelRobotCommands.add(btnSetBounds);
220
    panelRobotCommands.add(btnClearBounds);
221
                panelRobotCommands.add(btnCommand_MoveTo);
222
                panelRobotCommands.add(btnCommand_MoveAll);
223
                //panelRobotCommands.add(btnCommand_StopTask);
224
                //panelRobotCommands.add(btnCommand_ResumeTask);
225
                panelRobotControl = new JPanel();
226
                panelRobotControl.setLayout(new GridLayout(2,1));
227
                panelRobotControl.add(panelRobotDirection);
228
                panelRobotControl.add(panelRobotCommands);
229

    
230
                // Task Manager
231
                panelTaskManager = new JPanel();
232
                panelTaskManager.setLayout(new BorderLayout());
233
                taskListModel = new DefaultListModel();
234
                taskList = new JList(taskListModel);
235
                taskList.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
236
                taskList.setSelectedIndex(0);
237
                spTaskManager = new JScrollPane(taskList);
238
                panelTaskManagerControls = new JPanel();
239
                panelTaskManagerControls.setLayout(new GridLayout(1,4));
240
                panelTaskManagerControlsPriority = new JPanel();
241
                panelTaskManagerControlsPriority.setLayout(new GridLayout(1,2));
242
                btnAddTask = new JButton("Add...");
243
                btnRemoveTask = new JButton("Remove");
244
                btnMoveTaskUp = new JButton("^");
245
                btnMoveTaskDown = new JButton("v");
246
                btnUpdateTasks = new JButton("Update");
247
                panelTaskManagerControlsPriority.add(btnMoveTaskUp);
248
                panelTaskManagerControlsPriority.add(btnMoveTaskDown);
249
                panelTaskManagerControls.add(btnAddTask);
250
                panelTaskManagerControls.add(btnRemoveTask);
251
                panelTaskManagerControls.add(btnUpdateTasks);
252
                panelTaskManagerControls.add(panelTaskManagerControlsPriority);
253
                panelTaskManager.add(spTaskManager, BorderLayout.CENTER);
254
                panelTaskManager.add(panelTaskManagerControls, BorderLayout.SOUTH);
255
                panelTaskManager.add(new JLabel("Current Task Queue"), BorderLayout.NORTH);
256
                taskAddWindow = new TaskAddWindow();
257

    
258
                // Main control mechanism
259
                tabPaneControl = new JTabbedPane(JTabbedPane.TOP);
260
                tabPaneControl.setFont(new Font("arial", Font.PLAIN, 14));
261
                tabPaneControl.setPreferredSize(new Dimension(ColonetConstants.VECTOR_CONTROLLER_WIDTH, 0));
262
                tabPaneControl.addTab("Connection", panelServerInterface);
263
                tabPaneControl.addTab("Robots", panelRobotControl);
264
                //tabPaneControl.addTab("Tasks", panelTaskManager);
265

    
266
                // Put all elements in the ContentPane
267
                this.getContentPane().setLayout(new BorderLayout());
268
                this.getContentPane().add(tabPaneMain, BorderLayout.CENTER);
269
                this.getContentPane().add(tabPaneControl, BorderLayout.EAST);
270
                this.setVisible(true);
271
                
272
                // Disable components before connecting
273
                btnConnect.setText("Connect");
274
    lblConnectionStatus.setText("Status: Disconnected");
275
    setComponentsEnabled(false);
276

    
277
                /* Add all listeners here */
278
                // Task Management
279
                btnAddTask.addActionListener(this);
280
                btnRemoveTask.addActionListener(this);
281
                btnMoveTaskUp.addActionListener(this);
282
                btnMoveTaskDown.addActionListener(this);
283
                btnUpdateTasks.addActionListener(this);
284
                // Robot Control
285
                btnF.addActionListener(this);
286
                btnB.addActionListener(this);
287
                btnL.addActionListener(this);
288
                btnR.addActionListener(this);
289
                btnF.addKeyListener(this);
290
                btnB.addKeyListener(this);
291
                btnL.addKeyListener(this);
292
                btnR.addKeyListener(this);
293
                btnActivate.addActionListener(this);
294
                btnActivate.addKeyListener(this);
295
                btnCommand_MoveTo.addActionListener(this);
296
                btnCommand_MoveAll.addActionListener(this);
297
                btnCommand_StopTask.addActionListener(this);
298
                btnCommand_ResumeTask.addActionListener(this);
299
                btnCommand_ChargeNow.addActionListener(this);
300
                btnCommand_StopCharging.addActionListener(this);
301
                // Other
302
                btnConnect.addActionListener(this);
303
                btnGetXBeeIDs.addActionListener(this);
304
                btnAssignID.addActionListener(this);
305
                btnLocateStation.addActionListener(this);
306
    btnSetBounds.addActionListener(this);
307
    btnClearBounds.addActionListener(this);
308
                panelWebcam.addMouseListener(this);
309
                panelWebcam.addMouseMotionListener(this);
310
        }
311
        
312
        public ColonetServerInterface getCSI () {
313
          return csi;
314
        }
315

    
316
        public void paint (Graphics g) {
317
                super.paint(g);
318
        }
319

    
320
        public void update (Graphics g) {
321
                paint(g);
322
        }
323

    
324
        /**
325
        * Gets the JTextArea used for displaying debugging information. 
326
        *
327
        * @return the JTextArea where debugging information can be displayed.
328
        */
329
        public JTextArea getInfoPanel () {
330
                return txtInfo;
331
        }
332

    
333
        /**
334
        * Parses a String containing BOM matrix information.
335
        * The ColonetServerInterface receives lines of the BOM matrix.        (For encoding
336
        * information, see the ColonetServerInterface documentation.)         The entire matrix is passed
337
        * to the client when requested. This method takes a string of the form
338
        * "[command code] [command code] [number of robots] [data0] [data1] ..."
339
        * with tokens separated by spaces and containing no brackets.
340
        * The [command code]s are predefined values identifying this String as a BOM data
341
        * String, [number of robots] is an integer, and the values that follow are
342
        * the sensor readings of the robots in order, starting with robot 0.        Only [number of robots]^2
343
        * data entries will be read.        The matrix values are saved locally until the next String is parsed.
344
        *
345
        *
346
        * @param line the String containing BOM matrix information.
347
        * @throws ArrayIndexOutOfBoundsException if there are fewer than [number of robots]^2 data entries in the String
348
        */
349
        public void parseMatrix (String line) {
350
                txtInfo.setText("");
351
                String [] str = line.split(" ");
352
                int num = Integer.parseInt(str[2]);
353
                for (int i = 0; i < num; i++) {
354
                        for (int j = 0; j < num; j++) {
355
                                String next = str[3 + i*num + j];
356
                                if (next.equals("-1")) {
357
                                        txtInfo.append("-");
358
                                } else {
359
                                        txtInfo.append(next);
360
                                }
361

    
362
                                if (j < num - 1) {
363
                                        txtInfo.append(" ");
364
                                }
365
                        }
366

    
367
                        if (i < num - 1) {
368
                                txtInfo.append("\n");
369
                        }
370
                }
371
                repaint();
372
        }
373

    
374
        public void connect () {
375
    if (csi != null)
376
      return;
377
    csi = new ColonetServerInterface(self);
378
    csi.connect(txtHost.getText(), txtPort.getText());
379
    if (!csi.isReady()) {
380
      csi = null;
381
      return;
382
    }
383
    webcamLoader = new WebcamLoader(self);
384
    dataUpdater = new DataUpdater();
385
    dataUpdater.start();
386
    webcamLoader.start();
387
    Runnable r = new Runnable() {
388
      public void run () {
389
        btnConnect.setText("Disconnect");
390
        lblConnectionStatus.setText("Status: Connected");
391
        setComponentsEnabled(true);
392
      }
393
    };
394
    SwingUtilities.invokeLater(r);
395
        }
396

    
397
        public void disconnect () {
398
    try {
399
        dataUpdater.interrupt();
400
    } catch (Exception e) {
401
    }
402
    csi = null;
403
                Runnable r = new Runnable() {
404
      public void run () {
405
                    btnConnect.setText("Connect");
406
              lblConnectionStatus.setText("Status: Disconnected");
407
        setComponentsEnabled(false);
408
        boundary = null;
409
        station = null;
410
            }
411
                };
412
                SwingUtilities.invokeLater(r);
413
        }
414
        
415
        private void setComponentsEnabled (boolean enabled) {
416
    btnF.setEnabled(enabled);
417
    btnB.setEnabled(enabled);
418
    btnL.setEnabled(enabled);
419
    btnR.setEnabled(enabled);
420
    btnActivate.setEnabled(enabled);
421
    btnAssignID.setEnabled(enabled);
422
    btnLocateStation.setEnabled(enabled);
423
    btnSetBounds.setEnabled(enabled);
424
    btnClearBounds.setEnabled(enabled);
425
    btnCommand_ChargeNow.setEnabled(enabled);
426
    btnCommand_StopCharging.setEnabled(enabled);
427
    btnCommand_MoveTo.setEnabled(enabled);
428
    btnCommand_MoveAll.setEnabled(enabled);
429
        }
430

    
431
        /**
432
        * Parses a String containing a task queue update.
433
        * Format is currently not specified.
434
        * This method currently does nothing.
435
        *
436
        * @param line the String containing task queue update information.
437
        */
438
        public void parseQueue (String line) {
439

    
440
        }
441

    
442
        /**
443
        * Parses a String containing XBee ID values.
444
        * The ColonetServerInterface receives Strings of XBee information.        (For encoding
445
        * information, see the ColonetServerInterface documentation.)         This method takes
446
        * a string of the form "[command code] [command code] [number of robots] [id0] [id1] ..."
447
        * with tokens separated by spaces and containing no brackets.
448
        * The [command code]s are predefined values identifying this String as an XBee
449
        * ID String, [number of robots] is an integer, and the values that follow are
450
        * the IDs of the robots in order, starting with robot 0.        Only [number of robots]
451
        * will be read.         The ID values are saved locally until the next String is parsed.
452
        * The purpose of having this list is to ensure that robots are properly identified for control purposes.
453
        * This keeps robot identification consistent between sessions and prevents arbitrary assignment.
454
        *
455
        * @param line the String containing XBee ID information.
456
        * @throws ArrayIndexOutOfBoundsException if there are fewer than [number of robots] IDs in the String
457
        * @see ColonetServerInterface#sendXBeeIDRequest()
458
        */
459
        public void parseXBeeIDs (String line) {
460
                String [] str = line.split(" ");
461
                int num = Integer.parseInt(str[2]);
462
                xbeeID = new int[num];
463
                for (int i = 0; i < num; i++) {
464
                        xbeeID[i] = Integer.parseInt(str[i+3]);
465
                }
466
        }
467

    
468
        /**
469
        * Parses a String containing battery information.
470
        * The ColonetServerInterface receives Strings of battery information.         (For encoding
471
        * information, see the ColonetServerInterface documentation.)         This method takes
472
        * a string of the form "[command code] [command code] [robot ID] [value]"
473
        * with tokens separated by spaces and containing no brackets.
474
        * The [command code]s are predefined values identifying this String as a battery
475
        * information String, [robot ID] is an integer, and [value] is a battery measurement.
476
        * This updates the batery information for a single robot.
477
        *
478
        *
479
        * @param line the String containing battery information.
480
        * @see ColonetServerInterface#sendBatteryRequest(int)
481
        */
482
        public void parseBattery (String line) {
483
          int botNum, level;
484
          try {
485
                    String [] str = line.split(" ");
486
                    botNum = Integer.parseInt(str[2]);
487
                    level = Integer.parseInt(str[3]);
488
                } catch (NumberFormatException e) {
489
                    System.out.println("Could not parse battery update");
490
                    return;
491
                }
492

    
493
                RobotIcon r = robotIcons.get(botNum);
494
                if (r != null) {
495
                    r.setBattery(batteryIcon.convert(level));  // set contextual battery meter
496
                    batteryIcon.setLevel(level);             // set solo battery meter
497
                }
498
                super.repaint();
499
        }
500

    
501
        /**
502
        * Parses a String containing visual robot position information along with
503
        * canonical ID assignments.
504
        */
505
        public void parsePositions (String line) {
506
                String [] str = line.split(" ");
507
                RobotList newList = new RobotList();
508

    
509
                for (int i = 2; i < str.length; i+=3) {
510
                        int id = Integer.parseInt(str[i]);
511
                        int x = Integer.parseInt(str[i+1]);
512
                        int y = Integer.parseInt(str[i+2]);
513
                        RobotIcon newIcon = new RobotIcon(id, x, y);
514
                        // Save previous robot information
515
                        RobotIcon oldIcon = robotIcons.get(id);
516
                        if (oldIcon != null) {
517
                    newIcon.setBattery(oldIcon.getBattery());
518
                                newIcon.setDestination(oldIcon.getDestination());
519
                        }
520
                        if (newIcon.getID() >= 0) {
521
                                newIcon.setColor(Color.GREEN);
522
                        }
523
                        newList.put(id, newIcon);
524
                }
525
                robotIcons = newList;
526
                repaint();
527
        }
528
  
529
  /**
530
    * Parses a message that indicates a robot has reached its destination
531
   */
532
  public void parseMoveUpdate (String line) {
533
    System.out.println("Got move update: " + line);
534
    String [] str = line.split(" ");
535
    int id = Integer.parseInt(str[1]);
536
    robotIcons.get(id).clearDestination();
537
  }
538
        
539
        /**
540
        * Set the ID of the selected robot
541
        */
542
        public void setSelectedBot (int id) {
543
            this.selectedBot = id;
544
            lblSelected.setText("" + this.selectedBot);
545
        }
546
        
547
        /**
548
        * Returns the ID of the selected robot
549
        */
550
        public int getSelectedBot () {
551
            return this.selectedBot;
552
        }
553

    
554
        //
555
        // MouseListener methods
556
        //
557
        public void mousePressed(MouseEvent e) {
558
                //Start a new Thread to handle the MouseEvent
559
                (new MouseHandler(e)).start();
560
        }
561
        public void mouseExited(MouseEvent e) {
562
        }
563
        public void mouseEntered(MouseEvent e) {
564
        }
565
        public void mouseReleased(MouseEvent e) {
566
    (new MouseHandler(e)).start();
567
        }
568
        public void mouseClicked(MouseEvent e) {
569
        }
570
        public void mouseDragged(MouseEvent e) {
571
    (new MouseHandler(e)).start();
572
        }
573
        public void mouseMoved(MouseEvent e) {
574
        }
575

    
576
        //
577
        // KeyListener methods
578
        //
579
        public void keyPressed (KeyEvent e) {
580
                //Start a new Thread to handle the KeyEvent
581
                (new KeyHandler(e)).start();
582
                repaint();
583
        }
584
        public void keyReleased (KeyEvent e) {
585
        }
586
        public void keyTyped (KeyEvent e) {
587
        }
588

    
589
        //
590
        // ActionListener method
591
        //
592
        public void actionPerformed (ActionEvent e) {
593
                // Start a new Thread to handle the ActionEvent
594
                (new ActionHandler(e)).start();
595
                repaint();
596
        }
597

    
598
        class MouseHandler extends Thread {
599
                MouseEvent e;
600

    
601
                public MouseHandler (MouseEvent event) {
602
                        super("MouseHandler");
603
                        this.e = event;
604
                }
605

    
606
                public void run () {
607
                        Point pt = panelWebcam.convertClick(e);
608

    
609
                        // If we are selecting a waypoint (destination) for a specific bot
610
                        if (setWaypoint && setWaypointID >= 0 && e.getID() == MouseEvent.MOUSE_PRESSED) {
611
                          // If the user clicks outside the boundary, do nothing
612
                          if (boundary.isActive() && !boundary.contains(e.getX(), e.getY())) {
613
                            return;
614
                          }
615
                                setWaypoint = false;
616
                                panelWebcam.setCursor(new Cursor(Cursor.DEFAULT_CURSOR));
617
                                if (selectedBot < 0) {
618
                                        return;
619
                                }
620
                                
621
                                RobotIcon r = robotIcons.get(selectedBot);
622
                                if (r != null) {
623
                                        r.setDestination(pt);
624
                                        if (csi != null) {
625
                                                csi.sendAbsoluteMove(r.getID(), pt.x, pt.y);
626
                                        }
627
                                }
628
                                return;
629
                        }
630
                        
631
                        // If we are ordering a charge (right-click on a station)
632
                        if ((e.getButton() == MouseEvent.BUTTON2 || e.getButton() == MouseEvent.BUTTON3)  
633
          && e.getID() == MouseEvent.MOUSE_PRESSED
634
          && station != null && station.contains(e.getX(), e.getY())) {
635
        if (selectedBot >= 0) {
636
          RobotIcon r = robotIcons.get(selectedBot);
637
          if (r != null) {
638
            Point p = panelWebcam.convertPoint(station.getX(), station.getY());
639
            r.setDestination(p);
640
          }
641
          csi.sendRecharge(selectedBot);
642
        }
643
                                return;
644
                        }
645

    
646
                        // Right-click also means we are moving a robot
647
                        if ((e.getButton() == MouseEvent.BUTTON2 || e.getButton() == MouseEvent.BUTTON3)  
648
          && e.getID() == MouseEvent.MOUSE_PRESSED) {
649
        // If the user clicks outside the boundary, do nothing
650
                          if (boundary.isActive() && !boundary.contains(e.getX(), e.getY())) {
651
                            return;
652
                          }
653
                                if (selectedBot < 0) {
654
                                        return;
655
                                }
656
                                RobotIcon r = robotIcons.get(selectedBot);
657
                                if (r != null) {
658
                                        r.setDestination(pt);
659
                                        if (csi != null) {
660
                                                csi.sendAbsoluteMove(r.getID(), pt.x, pt.y);
661
                                        }
662
                                }
663
                                return;
664
                        }
665

    
666
                        // If we are setting all waypoints
667
                        if (setWaypoint && e.getID() == MouseEvent.MOUSE_PRESSED) {
668
                          // If the user clicks outside the boundary, do nothing
669
                          if (boundary.isActive() && !boundary.contains(e.getX(), e.getY())) {
670
                            return;
671
                          }
672
                                setWaypoint = false;
673
                                panelWebcam.setCursor(new Cursor(Cursor.DEFAULT_CURSOR));
674
                                for (Map.Entry<Integer,RobotIcon> entry : robotIcons.entrySet()) {
675
                                        RobotIcon r = entry.getValue();
676
                                        r.setDestination(pt);
677
          //TODO: send absolute move commands
678
                                }
679
                                return;
680
                        }
681
      
682
      // If we are drawing a boundary rectangle
683
      if (boundary != null && boundary.isSetting()) {
684
        // Begin 
685
        if (e.getID() == MouseEvent.MOUSE_PRESSED) {
686
          boundary.panel_p1 = new Point(e.getX(), e.getY());
687
          boundary.panel_p2 = new Point(e.getX(), e.getY());
688
          boundary.img_p1 = new Point(pt.x, pt.y);
689
        }
690
        // Resize
691
        else if (e.getID() == MouseEvent.MOUSE_DRAGGED){
692
          boundary.panel_p2 = new Point(e.getX(), e.getY());
693
        }
694
        // Finish
695
        else if (e.getID() == MouseEvent.MOUSE_RELEASED){
696
          boundary.panel_p2 = new Point(e.getX(), e.getY());
697
          boundary.img_p2 = new Point(pt.x, pt.y);
698
          panelWebcam.setCursor(new Cursor(Cursor.DEFAULT_CURSOR));
699
          boundary.set = false;
700
          boundary.active = true;
701
          boundary.sendToServer();
702
        }
703
        return;
704
      }
705
      
706
      // If we are locating a charging station
707
      if (setStation) {
708
        setStation = false;
709
        station = new ChargingStation(e.getX(), e.getY());
710
        station.setActive(true);
711
        return;
712
      }
713

    
714
                        // Otherwise, we are selecting a bot, or doing nothing
715
                        RobotIcon r = robotIcons.getBoundingIcon(pt);
716
      if (r != null)
717
      {
718
        selectedBot = r.getID();
719
        lblSelected.setText("" + selectedBot);
720
      }
721
                }
722
        }
723

    
724
        class KeyHandler extends Thread {
725
                KeyEvent e;
726

    
727
                public KeyHandler (KeyEvent event) {
728
                        super("KeyHandler");
729
                        this.e = event;
730
                }
731

    
732
                public void run () {
733
                        int code = e.getKeyCode();
734
                        if (code == KeyEvent.VK_UP) {
735
                                vectorController.setMaxForward();
736
                                vectorController.sendToServer();
737
                        } else if (code == KeyEvent.VK_DOWN) {
738
                                vectorController.setMaxReverse();
739
                                vectorController.sendToServer();
740
                        } else if (code == KeyEvent.VK_LEFT) {
741
                                vectorController.setMaxLeft();
742
                                vectorController.sendToServer();
743
                        } else if (code == KeyEvent.VK_RIGHT) {
744
                                vectorController.setMaxRight();
745
                                vectorController.sendToServer();
746
                        } else if (code == KeyEvent.VK_S) {
747
                                vectorController.setZero();
748
                                vectorController.sendToServer();
749
                        }
750
                }
751
        }
752

    
753
        class ActionHandler extends Thread {
754
                ActionEvent e;
755

    
756
                public ActionHandler (ActionEvent event) {
757
                        super("ActionHandler");
758
                        this.e = event;
759
                }
760

    
761
                public void run () {
762
                        Object source = e.getSource();
763

    
764
                        // General Actions
765
                        if (source == btnConnect) {
766
                                if (csi == null) {
767
                                        connect();
768
                                } else {
769
                                        disconnect();
770
                                }
771
                        } else if (source == btnGetXBeeIDs) {
772
                                csi.sendXBeeIDRequest();
773
                        } else if (source == btnAssignID) {
774
                                String message;
775
                                if (selectedBot < 0) {
776
                                        message = "That robot is unidentified. Please specify its ID.";
777
                                } else {
778
                                        message = "That robot has ID " + selectedBot + ". You may reassign it now.";
779
                                }
780
                                String result = JOptionPane.showInputDialog(self, message, "Robot Identification", JOptionPane.QUESTION_MESSAGE);
781
                                if (result == null) {
782
                                        return;
783
                                }
784
                                int newID = -1;
785
                                try {
786
                                        newID = Integer.parseInt(result);
787
                                } catch (Exception ex) {
788
                                        csi.warn("Invalid ID.");
789
                                        return;
790
                                }
791
                                // Assign new ID and update display
792
                                if (csi != null) {
793
                                        csi.sendIDAssignment(selectedBot, newID);
794
                                }
795
                                selectedBot = newID;
796
                                lblSelected.setText("" + newID);
797
                  } else if (source == btnLocateStation) {
798
                    setStation = true;
799
                        } else if (source == btnF) { // Robot Movement Controls
800
                                vectorController.setMaxForward();
801
                                vectorController.sendToServer();
802
                                robotIcons.get(selectedBot).clearDestination();
803
                        } else if (source == btnB) {
804
                                vectorController.setMaxReverse();
805
                                vectorController.sendToServer();
806
                                robotIcons.get(selectedBot).clearDestination();
807
                        } else if (source == btnL) {
808
                                vectorController.setMaxLeft();
809
                                vectorController.sendToServer();
810
                                robotIcons.get(selectedBot).clearDestination();
811
                        } else if (source == btnR) {
812
                                vectorController.setMaxRight();
813
                                vectorController.sendToServer();
814
                                robotIcons.get(selectedBot).clearDestination();
815
                        } else if (source == btnActivate) {
816
                                vectorController.setZero();
817
                                vectorController.sendToServer();
818
                                if (selectedBot >= 0)
819
                                {
820
                                        robotIcons.get(selectedBot).clearDestination();
821
                                }
822
      } else if (source == btnSetBounds) {
823
        boundary.set = true;
824
        panelWebcam.setCursor(new Cursor(Cursor.CROSSHAIR_CURSOR));
825
      } else if (source == btnClearBounds) {
826
        boundary.active = false;
827
        boundary.set = false;
828
        boundary.img_p1 = new Point(-1, -1);
829
        boundary.img_p2 = new Point(-1, -1);
830
        boundary.panel_p1 = new Point(-1, -1);
831
        boundary.panel_p2 = new Point(-1, -1);
832
        boundary.sendToServer();
833
                        } else if (source == btnCommand_MoveTo) {
834
                                if (selectedBot < 0) {
835
                                        return;
836
                                }
837
                                panelWebcam.setCursor(new Cursor(Cursor.CROSSHAIR_CURSOR));
838
                                setWaypoint = true;
839
                                setWaypointID = selectedBot;
840
                        } else if (source == btnCommand_MoveAll) {
841
                                panelWebcam.setCursor(new Cursor(Cursor.CROSSHAIR_CURSOR));
842
                                setWaypoint = true;
843
                                setWaypointID = -1;
844
                        } else if (source == btnCommand_StopTask) {
845

    
846
                        } else if (source == btnCommand_ResumeTask) {
847

    
848
                        } else if (source == btnCommand_ChargeNow) {
849
                          if (selectedBot >= 0)
850
                            csi.sendRecharge(selectedBot);
851
                          else
852
          csi.sendRecharge(2);
853
                        } else if (source == btnCommand_StopCharging) {
854
                          if (selectedBot >= 0) {
855
                            robotIcons.get(selectedBot).clearDestination();
856
                            csi.sendRechargeStop(selectedBot);
857
                          } else {
858
                            robotIcons.get(2).clearDestination();
859
          csi.sendRechargeStop(2);
860
        }
861
                        } else if (source == btnAddTask) { // Queue Management
862
                                taskAddWindow.prompt();
863
                        } else if (source == btnRemoveTask) {
864
                                if (taskList.getSelectedIndex() >= 0) {
865
                                        csi.sendQueueRemove(taskList.getSelectedIndex());
866
                                }
867
                                csi.sendQueueUpdate();
868
                        } else if (source == btnMoveTaskUp) {
869
                                csi.sendQueueReorder(taskList.getSelectedIndex(), taskList.getSelectedIndex() - 1);
870
                                csi.sendQueueUpdate();
871
                        } else if (source == btnMoveTaskDown) {
872
                                csi.sendQueueReorder(taskList.getSelectedIndex(), taskList.getSelectedIndex() + 1);
873
                                csi.sendQueueUpdate();
874
                        } else if (source == btnUpdateTasks) {
875
                                csi.sendQueueUpdate();
876
                        }
877

    
878
                }
879
        }
880

    
881
        /*
882
        * DataUpdater thread.
883
        *        The purpose of this thread is to request data from the server at regular intervals.
884
        *        Types of data requested: XBee IDs, robot positions, battery levels.
885
        *
886
        */
887
        class DataUpdater extends Thread {
888
                final int DATAUPDATER_DELAY = 500;
889
                int count;
890
                public DataUpdater () {
891
                        super("Colonet DataUpdater");
892
                        count = 0;
893
                }
894

    
895
                public synchronized void run () {
896
                        while (true) {
897
                                try {
898
                                        if (csi != null && csi.isReady()) {
899
                                                // XBee ID Request
900
                                                if (count % 5 == 0) {
901
                                                        csi.sendXBeeIDRequest();
902
                                                }
903
                                                // Robot Position Request
904
                                                if (count % 1 == 0) {
905
                                                        csi.sendPositionRequest();
906
                                                }
907
                                                // Battery Request
908
                                                if (count % 30 == 0) {
909
                                                        for (Map.Entry<Integer,RobotIcon> entry : robotIcons.entrySet()) {
910
                                                                RobotIcon r = entry.getValue();
911
                                                                        int id = r.getID();
912
                                                                                if (id >= 0) {
913
                                                                                csi.sendBatteryRequest(id);
914
                                                                        }
915
                                                        }
916
                                                }
917
                                        }
918
                                        count++;
919
                                        if (count > 1000)
920
                                                count = 0;
921
                                        Thread.sleep(DATAUPDATER_DELAY);
922
                                } catch (InterruptedException e) {
923
                                        return;
924
                                } catch (NullPointerException e) {
925
                                    // This probably indicates that an object was destroyed by shutdown.
926
                                    return;
927
                                }
928
                        }
929
                }
930
        }
931

    
932

    
933
        /*
934
        * WebcamPanel class
935
        * Contains the webcam image
936
        */
937
        class WebcamPanel extends JPanel {
938

    
939
                volatile BufferedImage img;
940
                BufferedImage buffer;
941

    
942
                public WebcamPanel () {
943
                        super(true);
944
                }
945

    
946
                public synchronized void setImage (BufferedImage newimg) {
947
                        if (img != null) {
948
                                img.flush();
949
                        }
950
                        System.gc();
951
                        img = newimg;
952
                        repaint();
953
                }
954

    
955
                public synchronized void paint (Graphics g) {
956
                        if (img == null) {
957
                                return;
958
                        }
959

    
960
                        // Calculate scaling
961
                        int maxWidth = getWidth() - 2*ColonetConstants.WEBCAM_BORDER;
962
                        int maxHeight = getHeight() - 2*ColonetConstants.WEBCAM_BORDER;
963
                        double widthRatio = 1.0 * maxWidth / img.getWidth();
964
                        double heightRatio = 1.0 * maxHeight / img.getHeight();
965
                        double scale = 0;
966
                        int newWidth = 0;
967
                        int newHeight = 0;
968
                        int x = 0;
969
                        int y = 0;
970

    
971
                        if (widthRatio > heightRatio) {         //height is the limiting factor
972
                                scale = heightRatio;
973
                                newHeight = maxHeight;
974
                                newWidth = (int) (img.getWidth() * scale);
975
                                y = ColonetConstants.WEBCAM_BORDER;
976
                                x = (maxWidth - newWidth) / 2 + ColonetConstants.WEBCAM_BORDER;
977
                        } else {        //width is the limiting factor
978
                                scale = widthRatio;
979
                                newWidth = maxWidth;
980
                                newHeight = (int) (img.getHeight() * scale);
981
                                x = ColonetConstants.WEBCAM_BORDER;
982
                                y = (maxHeight - newHeight) / 2 + ColonetConstants.WEBCAM_BORDER;
983
                        }
984

    
985
                        // Draw image onto the buffer
986
                        buffer = new BufferedImage(getWidth(), getHeight(), BufferedImage.TYPE_INT_RGB);
987
                        Graphics2D bufferedGraphics = (Graphics2D)buffer.getGraphics();
988
                        bufferedGraphics.setColor(Color.GRAY);
989
                        bufferedGraphics.fillRect(0, 0, this.getWidth(), this.getHeight());
990
                        Image imgScaled = img.getScaledInstance(newWidth, newHeight, Image.SCALE_FAST);
991
                        bufferedGraphics.drawImage(imgScaled, x, y, this);
992

    
993
      // Draw boundary
994
      if (boundary != null && (boundary.isSetting() || boundary.isActive())) {
995
         bufferedGraphics.setColor(Color.BLUE);
996
         int width = boundary.panel_p2.x - boundary.panel_p1.x;
997
         int height = boundary.panel_p2.y - boundary.panel_p1.y;
998
         bufferedGraphics.drawRect(boundary.panel_p1.x, boundary.panel_p1.y, width, height);
999
      }
1000
      
1001
      // Draw charging station
1002
      if (station != null && station.isActive()) {
1003
        bufferedGraphics.setStroke(new BasicStroke(2));
1004
        bufferedGraphics.setColor(Color.ORANGE);
1005
        int width = ColonetConstants.STATION_SIZE;
1006
        bufferedGraphics.drawRect(station.getX() - width/2, station.getY() - width/2, width, width);
1007
      }
1008

    
1009
                        // Draw Identifiers and battery levels
1010
                        if (robotIcons != null) {
1011
                                bufferedGraphics.setStroke(new BasicStroke(2));
1012
                                for (Map.Entry<Integer,RobotIcon> entry : robotIcons.entrySet()) {
1013
                                        RobotIcon r = entry.getValue();
1014
                                        bufferedGraphics.setColor(r.getColor());
1015
                                        // Identifier circle
1016
                                        int px = (int) (x + r.getLocation().x * scale);
1017
                                        int py = (int) (y + r.getLocation().y * scale);
1018
                                        int radius = ColonetConstants.ROBOT_RADIUS;
1019
                                        bufferedGraphics.drawOval(px-radius, py-radius, 2*radius, 2*radius);
1020
                                        // ID, if applicable
1021
                                        if (r.getID() > 0) {
1022
                                            bufferedGraphics.setFont(new Font("arial", Font.PLAIN, 36));
1023
                                            bufferedGraphics.drawString("" + r.getID(), px-10, py+10);
1024
                                        }
1025
                                        // Battery
1026
                                        if (r.getBattery() >= 0) {
1027
                                            int pixels = r.getBattery() * ColonetConstants.BATTERY_WIDTH / 100;
1028
                                                if (r.getBattery() > 50)
1029
                                                    bufferedGraphics.setColor(Color.GREEN);
1030
                                                else if (r.getBattery() > 25)
1031
                                                    bufferedGraphics.setColor(Color.YELLOW);
1032
                                                else
1033
                                                    bufferedGraphics.setColor(Color.RED);
1034
                                                bufferedGraphics.fillRect(px+20, py+20, pixels, ColonetConstants.BATTERY_HEIGHT);
1035
                                                bufferedGraphics.setColor(Color.BLACK);
1036
                                                bufferedGraphics.drawRect(px+20, py+20, ColonetConstants.BATTERY_WIDTH, ColonetConstants.BATTERY_HEIGHT);
1037
                                        }
1038
                                        // If the robot has a destination, draw the vector
1039
                                        if (r.hasDestination()) {
1040
                                                bufferedGraphics.drawLine(px, py, (int)(x + r.getDestination().x * scale), (int)(y + r.getDestination().y * scale));
1041
                                        }
1042
                                }
1043
                        }
1044

    
1045
                        // Identify currently-selected robot
1046
                        RobotIcon r = robotIcons.get(selectedBot);
1047
                        if (r != null) {
1048
                                int px = (int) (x + r.getLocation().x * scale);
1049
                                int py = (int) (y + r.getLocation().y * scale);
1050
                                int radius = ColonetConstants.ROBOT_RADIUS;
1051
                                bufferedGraphics.setColor(Color.BLACK);
1052
                                bufferedGraphics.drawOval(px-radius-6, py-radius-6, 2*radius+12, 2*radius+12);
1053
                        }
1054

    
1055
                        //Display buffered content
1056
                        g.drawImage(buffer, 0, 0, this);
1057
                }
1058

    
1059
                /**
1060
                * Convert a click on the webcam panel to a coordinate that is consistent with the
1061
                * original size of the image that the panel contains.
1062
                */
1063
                public Point convertClick (MouseEvent e) {
1064
                        if (img == null) {
1065
                                return new Point(e.getX(), e.getY());
1066
                        }
1067
                        return convertPoint(e.getX(), e.getY());
1068
                }
1069
                
1070
                /**
1071
                * Convert a point on the webcam panel to a coordinate that is consistent with the
1072
                * original size of the image that the panel contains.
1073
                */
1074
                public Point convertPoint (int panelx, int panely) {
1075

    
1076
                        // Calculate scaling
1077
                        int border = ColonetConstants.WEBCAM_BORDER;
1078
                        int clickx = panelx;
1079
                        int clicky = panely;
1080
                        int maxWidth = getWidth() - 2*border;
1081
                        int maxHeight = getHeight() - 2*border;
1082
                        double widthRatio = 1.0 * maxWidth / img.getWidth();
1083
                        double heightRatio = 1.0 * maxHeight / img.getHeight();
1084
                        double scale = 0;
1085
                        int newWidth = 0;
1086
                        int newHeight = 0;
1087
                        int px = 0;
1088
                        int py = 0;
1089

    
1090
                        if (widthRatio > heightRatio) {         //height is the limiting factor
1091
                                scale = heightRatio;
1092
                                newHeight = maxHeight;
1093
                                newWidth = (int) (img.getWidth() * scale);
1094
                                py = clicky - border;
1095
                                px = clickx - border - (maxWidth - newWidth) / 2;
1096
                        } else {        //width is the limiting factor
1097
                                scale = widthRatio;
1098
                                newWidth = maxWidth;
1099
                                newHeight = (int) (img.getHeight() * scale);
1100
                                px = clickx - border;
1101
                                py = clicky - border - (maxHeight - newHeight) / 2;
1102
                        }
1103
                        py = (int) (py / scale);
1104
                        px = (int) (px / scale);
1105
                        return new Point(px, py);
1106
                }
1107
                
1108
        }
1109

    
1110
        /*
1111
        * WebcamLoader class
1112
        * Handles the loading of the webcam image.
1113
        */
1114
        class WebcamLoader extends Thread
1115
        {
1116
                URL imagePath;
1117
                MediaTracker mt;
1118
                BufferedImage image;
1119
                Random rand;
1120

    
1121
                public WebcamLoader (JApplet applet)
1122
                {
1123
                        super("ColonetWebcamLoader");
1124
                        mt = new MediaTracker(applet);
1125
                        ImageIO.setUseCache(false);
1126
                        rand = new Random();
1127
                }
1128

    
1129
                public void run ()
1130
                {
1131
                        while (true) {
1132
                                try {
1133
                                        Thread.sleep(ColonetConstants.WEBCAM_DELAY);
1134
                                        if (image != null)
1135
                                                image.flush();
1136
                                        System.gc();
1137
                                        try {
1138
                                                imagePath = new URL(ColonetConstants.WEBCAM_PATH);
1139
                                        } catch (MalformedURLException e) {
1140
                                                System.out.println("Malformed URL: could not form URL from: [" + ColonetConstants.WEBCAM_PATH + "]\n");
1141
                                        }
1142
                                        image = ImageIO.read(imagePath);
1143
                                        // The MediaTracker waitForID pauses the thread until the image is loaded.
1144
                                        // We don't want to display a half-downloaded image.
1145
                                        mt.addImage(image, 1);
1146
                                        while(!mt.checkID(1))
1147
                                                Thread.sleep(20);
1148
                                        mt.removeImage(image);
1149
                                        // Save
1150
                                        panelWebcam.setImage(image);
1151
                                } catch (InterruptedException e) {
1152
                                        return;
1153
                                } catch (java.security.AccessControlException e) {
1154
                                        csi.warn("Could not load webcam.\n" + e);
1155
                                        return;
1156
                                } catch (IOException e) {
1157
                                        getInfoPanel().append("IOException while trying to load image.\n");
1158
                                }
1159
                        }
1160
                }
1161
        }
1162
  
1163
  class RobotBoundary {
1164
    public volatile boolean active, set;
1165
    public volatile Point img_p1, img_p2;
1166
    public volatile Point panel_p1, panel_p2;
1167
    
1168
    public RobotBoundary () {
1169
      active = false;
1170
      set = false;
1171
      img_p1 = new Point(-1,-1);
1172
      img_p2 = new Point(-1,-1);
1173
      panel_p1 = new Point(-1,-1); 
1174
      panel_p2 = new Point(-1,-1);
1175
    }
1176
    
1177
    public void sendToServer () {
1178
      if (csi != null) {
1179
        if (img_p1.x == -1 || img_p1.y == -1)
1180
          csi.sendBoundaryClear();
1181
        else
1182
          csi.sendBoundary(img_p1.x, img_p1.y, img_p2.x, img_p2.y);
1183
      }
1184
    }
1185
    
1186
    public boolean isActive () {
1187
      return active;
1188
    }
1189
    
1190
    public boolean isSetting () {
1191
      return set;
1192
    }
1193
    
1194
    /**
1195
    *  Returns a Rectangle designating the active area 
1196
    *  in the coordinate system of the JPanel.
1197
    *  Returns null if the boundary is not active.
1198
    */
1199
    public Rectangle toRectangle () {
1200
      if (!active)
1201
        return null;
1202
      return new Rectangle (panel_p1.x, panel_p1.y, (panel_p2.x-panel_p1.x), (panel_p2.y-panel_p1.y));
1203
    }
1204
    
1205
    /** 
1206
    *  Determines whether a coordinate in the coordinate
1207
    *  system of the JPanel is contained within the boundary.
1208
    *  If the boundary is not active, then this method 
1209
    *  returns false.
1210
    */
1211
    public boolean contains (int px, int py) {
1212
      Rectangle rect = this.toRectangle();
1213
      if (rect == null || !this.active)
1214
        return false;
1215
      return rect.contains(px, py);
1216
    }
1217
        
1218
  }
1219
  
1220
  class ChargingStation {
1221
    private int x, y;
1222
    private boolean active;
1223
    
1224
    public ChargingStation () {
1225
      super();
1226
    }
1227
    
1228
    public ChargingStation (int x, int y) {
1229
      this();
1230
      setLocation(x, y);
1231
    }
1232
    
1233
    public int getX () {
1234
      return x;
1235
    }
1236
    
1237
    public int getY () {
1238
      return y;
1239
    }
1240
    
1241
    /**
1242
    *  Sets the center of the charging station to
1243
    *  the specified coordinate point.
1244
    */
1245
    public void setLocation (int x, int y) {
1246
      this.x = x;
1247
      this.y = y;
1248
    }
1249
    
1250
    public boolean isActive () {
1251
      return active;
1252
    }
1253
    
1254
    public void setActive (boolean status) {
1255
      this.active = status;
1256
    }
1257
    
1258
    /**
1259
    *  Determines whether a coordinate is contained within
1260
    *  the graphical area of the charging station.
1261
    *  Returns false if the charging station is not active.
1262
    */
1263
    public boolean contains (int px, int py) {
1264
      if (!active)
1265
        return false;
1266
      int width = ColonetConstants.STATION_SIZE;
1267
      Rectangle rect = new Rectangle(this.x - width/2, this.y - width/2, width, width);
1268
      return rect.contains(px, py);
1269
    }
1270
  
1271
  }
1272

    
1273
}