All Downloads are FREE. Search and download functionalities are using the official Maven repository.

weka.gui.MemoryUsagePanel Maven / Gradle / Ivy

/*
 *   This program 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
 *   (at your option) any later version.
 *
 *   This program 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 this program.  If not, see .
 */

/*
 * MemoryUsagePanel.java
 * Copyright (C) 2008-2012 University of Waikato, Hamilton, New Zealand
 */

package weka.gui;

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.Graphics;
import java.awt.Point;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.Collections;
import java.util.Enumeration;
import java.util.Hashtable;
import java.util.Properties;
import java.util.Vector;

import javax.swing.JButton;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;

import weka.core.Memory;
import weka.core.Utils;
import weka.gui.visualize.VisualizeUtils;

/**
 * A panel for displaying the memory usage.
 * 
 * @author fracpete (fracpete at waikato dot ac dot nz)
 * @version $Revision: 10216 $
 */
public class MemoryUsagePanel extends JPanel {

  /** for serialization. */
  private static final long serialVersionUID = -4812319791687471721L;

  /**
   * Specialized thread for monitoring the memory usage.
   * 
   * @author fracpete (fracpete at waikato dot ac dot nz)
   * @version $Revision: 10216 $
   */
  protected class MemoryMonitor extends Thread {

    /** the refresh interval in msecs. */
    protected int m_Interval;

    /** whether the thread is still running. */
    protected boolean m_Monitoring;

    /**
     * default constructor.
     */
    public MemoryMonitor() {
      super();

      setInterval(1000); // TODO: via props file
    }

    /**
     * Returns the refresh interval in msecs.
     * 
     * @return returns the refresh interval
     */
    public int getInterval() {
      return m_Interval;
    }

    /**
     * Sets the refresh interval in msecs.
     * 
     * @param value the refresh interval
     */
    public void setInterval(int value) {
      m_Interval = value;
    }

    /**
     * Returns whether the thread is still running.
     * 
     * @return true if the thread is still running
     */
    public boolean isMonitoring() {
      return m_Monitoring;
    }

    /**
     * stops the monitoring thread.
     */
    public void stopMonitoring() {
      m_Monitoring = false;
    }

    /**
     * The run method.
     */
    @Override
    public void run() {
      m_Monitoring = true;

      while (m_Monitoring) {
        try {
          Thread.sleep(m_Interval);

          // update GUI
          if (m_Monitoring) {
            Runnable doUpdate = new Runnable() {
              @Override
              public void run() {
                update();
              }
            };
            SwingUtilities.invokeLater(doUpdate);
          }
        } catch (InterruptedException ex) {
          ex.printStackTrace();
        }
      }
    }

    /**
     * Updates the GUI.
     */
    protected void update() {
      double perc;
      Dimension size;

      // current usage
      perc = (double) m_Memory.getCurrent() / (double) m_Memory.getMax();
      perc = Math.round(perc * 1000) / 10;

      // tool tip
      setToolTipText("" + perc + "% used");

      // update history
      m_History.insertElementAt(perc, 0);
      size = getSize();
      while (m_History.size() > size.getWidth()) {
        m_History.remove(m_History.size() - 1);
      }

      // display history
      repaint();
    }
  }

  /** The name of the properties file. */
  protected static String PROPERTY_FILE = "weka/gui/MemoryUsage.props";

  /** Contains the properties. */
  protected static Properties PROPERTIES;

  /** the memory usage over time. */
  protected Vector m_History;

  /** for monitoring the memory usage. */
  protected Memory m_Memory;

  /** the thread for monitoring the memory usage. */
  protected MemoryMonitor m_Monitor;

  /** the button for running the garbage collector. */
  protected JButton m_ButtonGC;

  /** the threshold percentages to change color. */
  protected Vector m_Percentages;

  /** the corresponding colors for the thresholds. */
  protected Hashtable m_Colors;

  /** the default color. */
  protected Color m_DefaultColor;

  /** the background color. */
  protected Color m_BackgroundColor;

  /** the position for the dialog. */
  protected Point m_FrameLocation;

  /**
   * Loads the configuration property file (USE_DYNAMIC is FALSE) or determines
   * the classes dynamically (USE_DYNAMIC is TRUE)
   * 
   * @see #USE_DYNAMIC
   * @see GenericPropertiesCreator
   */
  static {
    // Allow a properties file in the current directory to override
    try {
      PROPERTIES = Utils.readProperties(PROPERTY_FILE);
      Enumeration keys = PROPERTIES.propertyNames();
      if (!keys.hasMoreElements()) {
        throw new Exception("Failed to read a property file for the "
          + "memory usage panel");
      }
    } catch (Exception ex) {
      JOptionPane.showMessageDialog(null,
        "Could not read a configuration file for the memory usage\n"
          + "panel. An example file is included with the Weka distribution.\n"
          + "This file should be named \"" + PROPERTY_FILE + "\" and\n"
          + "should be placed either in your user home (which is set\n"
          + "to \"" + System.getProperties().getProperty("user.home") + "\")\n"
          + "or the directory that java was started from\n",
        "MemoryUsagePanel", JOptionPane.ERROR_MESSAGE);
    }
  }

  /**
   * default constructor.
   */
  public MemoryUsagePanel() {
    super();

    // initializes members
    m_Memory = new Memory();
    m_History = new Vector();
    m_Percentages = new Vector();
    m_Colors = new Hashtable();

    // colors and percentages
    m_BackgroundColor = parseColor("BackgroundColor", Color.WHITE);
    m_DefaultColor = parseColor("DefaultColor", Color.GREEN);
    String[] percs = PROPERTIES.getProperty("Percentages", "70,80,90").split(
      ",");
    for (String perc2 : percs) {
      // do we have a color associated with percentage?
      if (PROPERTIES.getProperty(perc2) != null) {
        double perc;
        Color color;

        // try parsing the number
        try {
          perc = Double.parseDouble(perc2);
        } catch (Exception e) {
          System.err.println("MemoryUsagePanel: cannot parse percentage '"
            + perc2 + "' - ignored!");
          continue;
        }

        // try parsing the color
        color = parseColor(perc2, null);
        if (color == null) {
          continue;
        }

        // store color and percentage
        m_Percentages.add(perc);
        m_Colors.put(perc, color);
      } else {
        System.err
          .println("MemoryUsagePanel: cannot find color for percentage '"
            + perc2 + "' - ignored!");
      }
    }
    Collections.sort(m_Percentages);

    // layout
    setLayout(new BorderLayout());

    JPanel panel = new JPanel(new BorderLayout());
    add(panel, BorderLayout.EAST);

    m_ButtonGC = new JButton("GC");
    m_ButtonGC.setToolTipText("Runs the garbage collector.");
    m_ButtonGC.addActionListener(new ActionListener() {
      @Override
      public void actionPerformed(ActionEvent evt) {
        System.gc();
      }
    });
    panel.add(m_ButtonGC, BorderLayout.NORTH);

    // dimensions
    int height;
    int width;
    try {
      height = Integer.parseInt(PROPERTIES.getProperty("Height", ""
        + (int) m_ButtonGC.getPreferredSize().getHeight()));
      width = Integer.parseInt(PROPERTIES.getProperty("Width", "400"));
    } catch (Exception e) {
      System.err.println("MemoryUsagePanel: Problem parsing the dimensions - "
        + e);
      height = (int) m_ButtonGC.getPreferredSize().getHeight();
      width = 400;
    }
    setPreferredSize(new Dimension(width, height));

    // position
    int top;
    int left;
    try {
      top = Integer.parseInt(PROPERTIES.getProperty("Top", "0"));
      left = Integer.parseInt(PROPERTIES.getProperty("Left", "0"));
    } catch (Exception e) {
      System.err.println("MemoryUsagePanel: Problem parsing the position - "
        + e);
      top = 0;
      left = 0;
    }
    m_FrameLocation = new Point(left, top);

    // monitoring thread
    int interval;
    try {
      interval = Integer.parseInt(PROPERTIES.getProperty("Interval", "1000"));
    } catch (Exception e) {
      System.err
        .println("MemoryUsagePanel: Problem parsing the refresh interval - "
          + e);
      interval = 1000;
    }
    m_Monitor = new MemoryMonitor();
    m_Monitor.setInterval(interval);
    m_Monitor.setPriority(Thread.MAX_PRIORITY);
    m_Monitor.start();
  }

  /**
   * parses the color and returns the corresponding Color object.
   * 
   * @param prop the color property to read and parse
   * @param defValue the default color
   * @return the parsed color or the default color of the
   */
  protected Color parseColor(String prop, Color defValue) {
    Color result;
    Color color;
    String colorStr;

    result = defValue;

    try {
      colorStr = PROPERTIES.getProperty(prop);
      color = VisualizeUtils.processColour(colorStr, result);
      if (color == null) {
        throw new Exception(colorStr);
      }
      result = color;
    } catch (Exception e) {
      System.err.println("MemoryUsagePanel: cannot parse color '"
        + e.getMessage() + "' - ignored!");
    }

    return result;
  }

  /**
   * Returns whether the thread is still running.
   * 
   * @return true if the thread is still running
   */
  public boolean isMonitoring() {
    return m_Monitor.isMonitoring();
  }

  /**
   * stops the monitoring thread.
   */
  public void stopMonitoring() {
    m_Monitor.stopMonitoring();
  }

  /**
   * Returns the default position for the dialog.
   * 
   * @return the default position
   */
  public Point getFrameLocation() {
    return m_FrameLocation;
  }

  /**
   * draws the background image.
   * 
   * @param g the graphics context
   */
  @Override
  public void paintComponent(Graphics g) {
    int i;
    int n;
    int len;
    double scale;
    double perc;
    Color color;

    super.paintComponent(g);

    g.setColor(m_BackgroundColor);
    g.fillRect(0, 0, getWidth(), getHeight());
    scale = getHeight() / 100.0;
    for (i = 0; i < m_History.size(); i++) {
      perc = m_History.get(i);

      // determine color
      color = m_DefaultColor;
      for (n = m_Percentages.size() - 1; n >= 0; n--) {
        if (perc >= m_Percentages.get(n)) {
          color = m_Colors.get(m_Percentages.get(n));
          break;
        }
      }

      // paint line
      g.setColor(color);
      len = (int) Math.round(perc * scale);
      g.drawLine(i, getHeight() - 1, i, getHeight() - len);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy