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("
");
            while ( (line=in.readLine()) != null ) {
               text.append(line+"\n");
            }
            in.close();
            text.append("
"); 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 = "
"+// width="+this.getSize().getWidth()+">"+ ""+ ""+ ""+ ""+ ""+ ""+ "" + ""+ ""+ ""+ ""+ ""+ ""+ ""+ "" + ""+ ""+ "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + "" + ""+ "" + "
  Operand Key for Example Instructions  
label, targetany textual label
$t1, $t2, $t3any integer register
$f2, $f4, $f6even-numbered floating point register
$f0, $f1, $f3any floating point register
$8any Coprocessor 0 register
1condition flag (0 to 7)
10unsigned 5-bit integer (0 to 31)
-100signed 16-bit integer (-32768 to 32767)
100unsigned 16-bit integer (0 to 65535)
100000signed 32-bit integer (-2147483648 to 2147483647)
Load & Store addressing mode, basic instructions
-100($t2)sign-extended 16-bit integer added to contents of $t2
Load & Store addressing modes, pseudo instructions
($t2)contents of $t2
-100signed 16-bit integer
100unsigned 16-bit integer
100000signed 32-bit integer
100($t2)zero-extended unsigned 16-bit integer added to contents of $t2
100000($t2)signed 32-bit integer added to contents of $t2
label32-bit address of label
label($t2)32-bit address of label added to contents of $t2
label+10000032-bit integer added to label's address
label+100000($t2)   sum of 32-bit integer, label's address, and contents of $t2
"; // 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 = "Unable to display requested document."; 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); } } } } }