Files
EMARS/mars/venus/HelpHelpAction.java
T
adolphenom 0da1c5dcca Source code of MARS Assembler
First commit of the 4.5 version (latest version available)
2014-12-21 12:49:28 +01:00

445 lines
23 KiB
Java

package mars.venus;
import mars.*;
import mars.assembler.*;
import mars.mips.instructions.*;
import java.util.*;
import java.io.*;
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.event.*;
import javax.swing.text.html.*;
/*
Copyright (c) 2003-2008, Pete Sanderson and Kenneth Vollmar
Developed by Pete Sanderson (psanderson@otterbein.edu)
and Kenneth Vollmar (kenvollmar@missouristate.edu)
Permission is hereby granted, free of charge, to any person obtaining
a copy of this software and associated documentation files (the
"Software"), to deal in the Software without restriction, including
without limitation the rights to use, copy, modify, merge, publish,
distribute, sublicense, and/or sell copies of the Software, and to
permit persons to whom the Software is furnished to do so, subject
to the following conditions:
The above copyright notice and this permission notice shall be
included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR
ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
(MIT license, http://www.opensource.org/licenses/mit-license.html)
*/
/**
* Action for the Help -> Help menu item
*/
public class HelpHelpAction extends GuiAction {
public HelpHelpAction(String name, Icon icon, String descrip,
Integer mnemonic, KeyStroke accel, VenusUI gui) {
super(name, icon, descrip, mnemonic, accel, gui);
}
// ideally read or computed from config file...
private Dimension getSize() {
return new Dimension(800,600);
}
// Light gray background color for alternating lines of the instruction lists
static Color altBackgroundColor = new Color(0xEE,0xEE,0xEE);
/**
* Separates Instruction name descriptor from detailed (operation) description
* in help string.
*/
public static final String descriptionDetailSeparator = ":";
/**
* Displays tabs with categories of information
*/
public void actionPerformed(ActionEvent e) {
JTabbedPane tabbedPane = new JTabbedPane();
tabbedPane.addTab("MIPS", createMipsHelpInfoPanel());
tabbedPane.addTab("MARS", createMarsHelpInfoPanel());
tabbedPane.addTab("License", createCopyrightInfoPanel());
tabbedPane.addTab("Bugs/Comments", createHTMLHelpPanel("BugReportingHelp.html"));
tabbedPane.addTab("Acknowledgements", createHTMLHelpPanel("Acknowledgements.html"));
tabbedPane.addTab("Instruction Set Song", createHTMLHelpPanel("MIPSInstructionSetSong.html"));
// Create non-modal dialog. Based on java.sun.com "How to Make Dialogs", DialogDemo.java
final JDialog dialog = new JDialog(mainUI, "MARS "+Globals.version+" Help");
// assure the dialog goes away if user clicks the X
dialog.addWindowListener(
new WindowAdapter() {
public void windowClosing(WindowEvent e) {
dialog.setVisible(false);
dialog.dispose();
}
});
//Add a "close" button to the non-modal help dialog.
JButton closeButton = new JButton("Close");
closeButton.addActionListener(
new ActionListener() {
public void actionPerformed(ActionEvent e) {
dialog.setVisible(false);
dialog.dispose();
}
});
JPanel closePanel = new JPanel();
closePanel.setLayout(new BoxLayout(closePanel,BoxLayout.LINE_AXIS));
closePanel.add(Box.createHorizontalGlue());
closePanel.add(closeButton);
closePanel.add(Box.createHorizontalGlue());
closePanel.setBorder(BorderFactory.createEmptyBorder(0,0,5,5));
JPanel contentPane = new JPanel();
contentPane.setLayout(new BoxLayout(contentPane,BoxLayout.PAGE_AXIS));
contentPane.add(tabbedPane);
contentPane.add(Box.createRigidArea(new Dimension(0,5)));
contentPane.add(closePanel);
contentPane.setOpaque(true);
dialog.setContentPane(contentPane);
//Show it.
dialog.setSize(this.getSize());
dialog.setLocationRelativeTo(mainUI);
dialog.setVisible(true);
//////////////////////////////////////////////////////////////////
}
// Create panel containing Help Info read from html document.
private JPanel createHTMLHelpPanel(String filename) {
JPanel helpPanel = new JPanel(new BorderLayout());
JScrollPane helpScrollPane;
JEditorPane helpDisplay;
try {
InputStream is = this.getClass().getResourceAsStream(Globals.helpPath+filename);
BufferedReader in = new BufferedReader(new InputStreamReader(is));
String line;
StringBuffer text = new StringBuffer();
while ( (line=in.readLine()) != null ) {
text.append(line+"\n");
}
in.close();
helpDisplay = new JEditorPane("text/html",text.toString());
helpDisplay.setEditable(false);
helpDisplay.setCaretPosition(0); // assure top of document displayed
helpScrollPane = new JScrollPane(helpDisplay, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
helpDisplay.addHyperlinkListener(new HelpHyperlinkListener());
}
catch (Exception ie) {
helpScrollPane = new JScrollPane(
new JLabel("Error ("+ie+"): "+filename+" contents could not be loaded."));
}
helpPanel.add(helpScrollPane);
return helpPanel;
}
// Set up the copyright notice for display.
private JPanel createCopyrightInfoPanel() {
JPanel marsCopyrightInfo = new JPanel(new BorderLayout());
JScrollPane marsCopyrightScrollPane;
JEditorPane marsCopyrightDisplay;
try {
InputStream is = this.getClass().getResourceAsStream("/MARSlicense.txt");
BufferedReader in = new BufferedReader(new InputStreamReader(is));
String line;
StringBuffer text = new StringBuffer("<pre>");
while ( (line=in.readLine()) != null ) {
text.append(line+"\n");
}
in.close();
text.append("</pre>");
marsCopyrightDisplay = new JEditorPane("text/html",text.toString());
marsCopyrightDisplay.setEditable(false);
marsCopyrightDisplay.setCaretPosition(0); // assure top of document displayed
marsCopyrightScrollPane = new JScrollPane(marsCopyrightDisplay, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
}
catch (Exception ioe) {
marsCopyrightScrollPane = new JScrollPane(
new JLabel("Error: license contents could not be loaded."));
}
marsCopyrightInfo.add(marsCopyrightScrollPane);
return marsCopyrightInfo;
}
// Set up MARS help tab. Subtabs get their contents from HTML files.
private JPanel createMarsHelpInfoPanel() {
JPanel marsHelpInfo = new JPanel(new BorderLayout());
JTabbedPane tabbedPane = new JTabbedPane();
tabbedPane.addTab("Intro", createHTMLHelpPanel("MarsHelpIntro.html"));
tabbedPane.addTab("IDE", createHTMLHelpPanel("MarsHelpIDE.html"));
tabbedPane.addTab("Debugging", createHTMLHelpPanel("MarsHelpDebugging.html"));
tabbedPane.addTab("Settings", createHTMLHelpPanel("MarsHelpSettings.html"));
tabbedPane.addTab("Tools", createHTMLHelpPanel("MarsHelpTools.html"));
tabbedPane.addTab("Command", createHTMLHelpPanel("MarsHelpCommand.html"));
tabbedPane.addTab("Limits", createHTMLHelpPanel("MarsHelpLimits.html"));
tabbedPane.addTab("History", createHTMLHelpPanel("MarsHelpHistory.html"));
marsHelpInfo.add(tabbedPane);
return marsHelpInfo;
}
// Set up MIPS help tab. Most contents are generated from instruction set info.
private JPanel createMipsHelpInfoPanel() {
JPanel mipsHelpInfo = new JPanel(new BorderLayout());
String helpRemarksColor = "CCFF99";
// Introductory remarks go at the top as a label
String helpRemarks =
"<html><center><table bgcolor=\"#"+helpRemarksColor+"\" border=0 cellpadding=0>"+// width="+this.getSize().getWidth()+">"+
"<tr>"+
"<th colspan=2><b><i><font size=+1>&nbsp;&nbsp;Operand Key for Example Instructions&nbsp;&nbsp;</font></i></b></th>"+
"</tr>"+
"<tr>"+
"<td><tt>label, target</tt></td><td>any textual label</td>"+
"</tr><tr>"+
"<td><tt>$t1, $t2, $t3</tt></td><td>any integer register</td>" +
"</tr><tr>"+
"<td><tt>$f2, $f4, $f6</tt></td><td><i>even-numbered</i> floating point register</td>"+
"</tr><tr>"+
"<td><tt>$f0, $f1, $f3</tt></td><td><i>any</i> floating point register</td>"+
"</tr><tr>"+
"<td><tt>$8</tt></td><td>any Coprocessor 0 register</td>"+
"</tr><tr>"+
"<td><tt>1</tt></td><td>condition flag (0 to 7)</td>" +
"</tr><tr>"+
"<td><tt>10</tt></td><td>unsigned 5-bit integer (0 to 31)</td>"+
"</tr><tr>" +
"<td><tt>-100</tt></td><td>signed 16-bit integer (-32768 to 32767)</td>" +
"</tr><tr>" +
"<td><tt>100</tt></td><td>unsigned 16-bit integer (0 to 65535)</td>" +
"</tr><tr>" +
"<td><tt>100000</tt></td><td>signed 32-bit integer (-2147483648 to 2147483647)</td>" +
"</tr><tr>" +
"</tr><tr>" +
"<td colspan=2><b><i><font size=+1>Load & Store addressing mode, basic instructions</font></i></b></td>" +
"</tr><tr>" +
"<td><tt>-100($t2)</tt></td><td>sign-extended 16-bit integer added to contents of $t2</td>" +
"</tr><tr>" +
"</tr><tr>" +
"<td colspan=2><b><i><font size=+1>Load & Store addressing modes, pseudo instructions</font></i></b></td>" +
"</tr><tr>" +
"<td><tt>($t2)</tt></td><td>contents of $t2</td>" +
"</tr><tr>" +
"<td><tt>-100</tt></td><td>signed 16-bit integer</td>" +
"</tr><tr>" +
"<td><tt>100</tt></td><td>unsigned 16-bit integer</td>" +
"</tr><tr>" +
"<td><tt>100000</tt></td><td>signed 32-bit integer</td>" +
"</tr><tr>" +
"<td><tt>100($t2)</tt></td><td>zero-extended unsigned 16-bit integer added to contents of $t2</td>" +
"</tr><tr>" +
"<td><tt>100000($t2)</tt></td><td>signed 32-bit integer added to contents of $t2</td>" +
"</tr><tr>" +
"<td><tt>label</tt></td><td>32-bit address of label</td>" +
"</tr><tr>" +
"<td><tt>label($t2)</tt></td><td>32-bit address of label added to contents of $t2</td>" +
"</tr><tr>" +
"<td><tt>label+100000</tt></td><td>32-bit integer added to label's address</td>" +
"</tr><tr>" +
"<td><tt>label+100000($t2)&nbsp;&nbsp;&nbsp;</tt></td><td>sum of 32-bit integer, label's address, and contents of $t2</td>"+
"</tr>" +
"</table></center></html>";
// Original code: mipsHelpInfo.add(new JLabel(helpRemarks, JLabel.CENTER), BorderLayout.NORTH);
JLabel helpRemarksLabel = new JLabel(helpRemarks, JLabel.CENTER);
helpRemarksLabel.setOpaque(true);
helpRemarksLabel.setBackground(Color.decode("0x"+helpRemarksColor));
JScrollPane operandsScrollPane = new JScrollPane(helpRemarksLabel, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
JScrollPane.HORIZONTAL_SCROLLBAR_ALWAYS);
mipsHelpInfo.add(operandsScrollPane, BorderLayout.NORTH);
// Below the label is a tabbed pane with categories of MIPS help
JTabbedPane tabbedPane = new JTabbedPane();
tabbedPane.addTab("Basic Instructions", createMipsInstructionHelpPane("mars.mips.instructions.BasicInstruction"));
tabbedPane.addTab("Extended (pseudo) Instructions", createMipsInstructionHelpPane("mars.mips.instructions.ExtendedInstruction"));
tabbedPane.addTab("Directives", createMipsDirectivesHelpPane());
tabbedPane.addTab("Syscalls", createHTMLHelpPanel("SyscallHelp.html"));
tabbedPane.addTab("Exceptions", createHTMLHelpPanel("ExceptionsHelp.html"));
tabbedPane.addTab("Macros", createHTMLHelpPanel("MacrosHelp.html"));
operandsScrollPane.setPreferredSize(new Dimension((int)this.getSize().getWidth(), (int) (this.getSize().getHeight()*.2)));
operandsScrollPane.getVerticalScrollBar().setUnitIncrement(10);
tabbedPane.setPreferredSize(new Dimension((int)this.getSize().getWidth(), (int) (this.getSize().getHeight()*.6)));
JSplitPane splitsville = new JSplitPane(JSplitPane.VERTICAL_SPLIT, operandsScrollPane, tabbedPane);
splitsville.setOneTouchExpandable(true);
splitsville.resetToPreferredSizes();
mipsHelpInfo.add(splitsville);
//mipsHelpInfo.add(tabbedPane);
return mipsHelpInfo;
}
/////////////// Methods to construct MIPS help tabs from internal MARS objects //////////////
/////////////////////////////////////////////////////////////////////////////
private JScrollPane createMipsDirectivesHelpPane() {
Vector exampleList = new Vector();
String blanks = " "; // 12 blanks
Directives direct;
Iterator it = Directives.getDirectiveList().iterator();
while (it.hasNext()) {
direct = (Directives)it.next();
exampleList.add(direct.toString()
+ blanks.substring(0,Math.max(0,blanks.length()-direct.toString().length()))
+ direct.getDescription());
}
Collections.sort(exampleList);
JList examples = new JList(exampleList);
JScrollPane mipsScrollPane = new JScrollPane(examples,JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
examples.setFont(new Font("Monospaced",Font.PLAIN,12));
return mipsScrollPane;
}
////////////////////////////////////////////////////////////////////////////
private JScrollPane createMipsInstructionHelpPane(String instructionClassName) {
ArrayList instructionList = Globals.instructionSet.getInstructionList();
Vector exampleList = new Vector(instructionList.size());
Iterator it = instructionList.iterator();
Instruction instr;
String blanks = " "; // 24 blanks
Class instructionClass;
while (it.hasNext()) {
instr = (Instruction) it.next();
try {
if (Class.forName(instructionClassName).isInstance(instr)) {
exampleList.add(instr.getExampleFormat()
+ blanks.substring(0,Math.max(0,blanks.length()-instr.getExampleFormat().length()))
+ instr.getDescription());
}
}
catch (ClassNotFoundException cnfe) {
System.out.println(cnfe+" "+instructionClassName);
}
}
Collections.sort(exampleList);
JList examples = new JList(exampleList);
JScrollPane mipsScrollPane = new JScrollPane(examples,JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
examples.setFont(new Font("Monospaced",Font.PLAIN,12));
examples.setCellRenderer(new MyCellRenderer());
return mipsScrollPane;
}
private class MyCellRenderer extends JLabel implements ListCellRenderer {
// This is the only method defined by ListCellRenderer.
// We just reconfigure the JLabel each time we're called.
public Component getListCellRendererComponent(
JList list, // the list
Object value, // value to display
int index, // cell index
boolean isSelected, // is the cell selected
boolean cellHasFocus) // does the cell have focus
{
String s = value.toString();
setText(s);
if (isSelected) {
setBackground(list.getSelectionBackground());
setForeground(list.getSelectionForeground());
}
else {
setBackground((index % 2 == 0) ? altBackgroundColor : list.getBackground());
setForeground(list.getForeground());
}
setEnabled(list.isEnabled());
setFont(list.getFont());
setOpaque(true);
return this;
}
}
/*
* Determines MARS response when user click on hyperlink in displayed help page.
* The response will be to pop up a simple dialog with the page contents. It
* will not display URL, no navigation, nothing. Just display the page and
* provide a Close button.
*/
private class HelpHyperlinkListener implements HyperlinkListener {
JDialog webpageDisplay;
JTextField webpageURL;
private static final String cannotDisplayMessage =
"<html><title></title><body><strong>Unable to display requested document.</strong></body></html>";
public void hyperlinkUpdate(HyperlinkEvent e) {
if (e.getEventType() == HyperlinkEvent.EventType.ACTIVATED) {
JEditorPane pane = (JEditorPane) e.getSource();
if (e instanceof HTMLFrameHyperlinkEvent) {
HTMLFrameHyperlinkEvent evt = (HTMLFrameHyperlinkEvent)e;
HTMLDocument doc = (HTMLDocument)pane.getDocument();
doc.processHTMLFrameHyperlinkEvent(evt);
}
else {
webpageDisplay = new JDialog(mainUI, "Primitive HTML Viewer");
webpageDisplay.setLayout(new BorderLayout());
webpageDisplay.setLocation(mainUI.getSize().width/6, mainUI.getSize().height/6);
JEditorPane webpagePane;
try {
webpagePane = new JEditorPane(e.getURL());
}
catch (Throwable t) {
webpagePane = new JEditorPane("text/html",cannotDisplayMessage);
}
webpagePane.addHyperlinkListener(
new HyperlinkListener() {
public void hyperlinkUpdate(HyperlinkEvent e) {
if (e.getEventType() == HyperlinkEvent.EventType.ACTIVATED) {
JEditorPane pane = (JEditorPane) e.getSource();
if (e instanceof HTMLFrameHyperlinkEvent) {
HTMLFrameHyperlinkEvent evt = (HTMLFrameHyperlinkEvent)e;
HTMLDocument doc = (HTMLDocument)pane.getDocument();
doc.processHTMLFrameHyperlinkEvent(evt);
}
else {
try {
pane.setPage(e.getURL());
}
catch (Throwable t) {
pane.setText(cannotDisplayMessage);
}
webpageURL.setText(e.getURL().toString());
}
}
}
});
webpagePane.setPreferredSize(new Dimension(mainUI.getSize().width*2/3, mainUI.getSize().height*2/3));
webpagePane.setEditable(false);
webpagePane.setCaretPosition(0);
JScrollPane webpageScrollPane = new JScrollPane(webpagePane,
JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED,
JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
webpageURL = new JTextField(e.getURL().toString(), 50);
webpageURL.setEditable(false);
webpageURL.setBackground(Color.WHITE);
JPanel URLPanel= new JPanel(new FlowLayout(FlowLayout.LEFT, 4, 4));
URLPanel.add(new JLabel("URL: "));
URLPanel.add(webpageURL);
webpageDisplay.add(URLPanel, BorderLayout.NORTH);
webpageDisplay.add(webpageScrollPane);
JButton closeButton = new JButton("Close");
closeButton.addActionListener(
new ActionListener() {
public void actionPerformed(ActionEvent e) {
webpageDisplay.setVisible(false);
webpageDisplay.dispose();
}
});
JPanel closePanel = new JPanel();
closePanel.setLayout(new BoxLayout(closePanel,BoxLayout.LINE_AXIS));
closePanel.setBorder(BorderFactory.createEmptyBorder(0,0,5,5));
closePanel.add(Box.createHorizontalGlue());
closePanel.add(closeButton);
closePanel.add(Box.createHorizontalGlue());
webpageDisplay.add(closePanel,BorderLayout.SOUTH);
webpageDisplay.pack();
webpageDisplay.setVisible(true);
}
}
}
}
}