import java.awt.*; import java.awt.event.KeyEvent; import java.util.ArrayList; import java.util.List; import javax.accessibility.Accessible; import javax.accessibility.AccessibleContext; import javax.accessibility.AccessibleState; import javax.accessibility.AccessibleStateSet; import javax.swing.*; import javax.swing.plaf.nimbus.NimbusLookAndFeel; /* * @test * @key headful * @bug 8134116 * @summary JTabbedPane$Page.getBounds throws IndexOutOfBoundsException * @run main Bug8134116 */ public class Bug8134116 { private static volatile Exception exception = null; public static void main(String args[]) throws Exception { try { UIManager.setLookAndFeel(new NimbusLookAndFeel()); } catch (Exception e) { throw new RuntimeException(e); } SwingUtilities.invokeAndWait(() -> { JPanel panel0 = new JPanel(); JPanel panel2 = new JPanel(); BadPane badPane = new BadPane(); badPane.add("zero", panel0); badPane.add("one", null); // no component badPane.add("", panel2); // no title badPane.add("", null); // no component, no title // but give it that via a tabComponent JPanel tabComponent = new JPanel(); JLabel tabComponentLabel = new JLabel("three"); tabComponent.add(tabComponentLabel); badPane.setTabComponentAt(3, tabComponent); JFrame frame = new JFrame(); frame.add(badPane); frame.setSize(300, 300); frame.setVisible(true); try { AccessibleContext ac = badPane.getAccessibleContext(); Accessible page0 = ac.getAccessibleChild(0); if (page0 == null) { // Not something being tested, but checking anyway throw new RuntimeException("getAccessibleChild(0) is null"); } Accessible page1 = ac.getAccessibleChild(1); if (page1 == null) { // Not something being tested, but checking anyway throw new RuntimeException("getAccessibleChild(1) is null"); } Accessible page2 = ac.getAccessibleChild(2); Accessible page3 = ac.getAccessibleChild(3); // page0 - page3 are JTabbedPane.Page, a private inner class // and is an AccessibleContext // and implements Accessible and AccessibleComponent AccessibleContext pac0 = page0.getAccessibleContext(); AccessibleContext pac1 = page1.getAccessibleContext(); AccessibleContext pac2 = page2.getAccessibleContext(); AccessibleContext pac3 = page3.getAccessibleContext(); // test Page.getBounds // ensure no IndexOutOfBoundsException Rectangle r0 = pac0.getAccessibleComponent().getBounds(); // make sure second Bounds is different than first Rectangle r1 = pac1.getAccessibleComponent().getBounds(); if (r1.equals(r0)) { String msg = "Second tab should not have same bounds as first tab"; throw new RuntimeException(msg); } // test Page.getAccessibleStateSet // At this point page 0 is selected AccessibleStateSet accSS0 = pac0.getAccessibleStateSet(); if (!accSS0.contains(AccessibleState.SELECTED)) { String msg = "Empty title -> AccessibleState.SELECTED not set"; throw new RuntimeException(msg); } // select second tab badPane.setSelectedIndex(1); AccessibleStateSet accSS1 = pac1.getAccessibleStateSet(); if (!accSS1.contains(AccessibleState.SELECTED)) { String msg = "Second tab selected but AccessibleState.SELECTED not set"; throw new RuntimeException(msg); } // select third tab badPane.setSelectedIndex(2); AccessibleStateSet accSS2 = pac2.getAccessibleStateSet(); if (!accSS1.contains(AccessibleState.SELECTED)) { String msg = "Third tab selected but AccessibleState.SELECTED not set"; throw new RuntimeException(msg); } // select fourth tab badPane.setSelectedIndex(3); AccessibleStateSet accSS3 = pac3.getAccessibleStateSet(); if (!accSS1.contains(AccessibleState.SELECTED)) { String msg = "Fourth tab selected but AccessibleState.SELECTED not set"; throw new RuntimeException(msg); } // test Page.getAccessibleIndexInParent if (pac0.getAccessibleIndexInParent() == -1) { String msg = "Empty title -> negative AccessibleIndexInParent"; throw new RuntimeException(msg); } if (pac0.getAccessibleIndexInParent() != 0) { String msg = "first tab is not at index 0 in parent"; throw new RuntimeException(msg); } if (pac1.getAccessibleIndexInParent() != 1) { String msg = "second tab (null component) is not at index 1 in parent"; throw new RuntimeException(msg); } if (pac2.getAccessibleIndexInParent() != 2) { String msg = "third tab (empty title) string is not at index 2 in parent"; throw new RuntimeException(msg); } if (pac3.getAccessibleIndexInParent() != 3) { String msg = "fourth tab (empty title, null component, has tabComponent) string is not at index 3 in parent"; throw new RuntimeException(msg); } // test Page.getAccessibleName String accName = pac0.getAccessibleName(); if (!accName.equals("zero")) { String msg = "Empty title -> empty AccessibleName"; throw new RuntimeException(msg); } // test Page.getAccessibleName when component is null accName = pac1.getAccessibleName(); if (!accName.equals("one")) { String msg = "AccessibleName of null panel not 'one'"; throw new RuntimeException(msg); } // test Page.setDisplayedMnemonicIndex // Empty title -> IllegalArgumnetException badPane.setDisplayedMnemonicIndexAt(0, 1); // test Page.updateDisplayedMnemonicIndex badPane.setMnemonicAt(0, KeyEvent.VK_Z); if (badPane.getDisplayedMnemonicIndexAt(0) == -1) { String msg="Empty title -> getDisplayedMnemonicIndexAt failure"; throw new RuntimeException(msg); } } catch (Exception e) { exception = e; } }); if (exception != null) { System.out.println("Test failed: " + exception.getMessage()); throw exception; } else { System.out.println("Test passed."); } } // The following is likely what is being done in Burp Suite // https://portswigger.net/burp/ which fails in the same way, i.e. the // pages List in JTabbedPane is not being managed properly and thus // Page.title is "" for each page. The overridden insertTab manages titles // in the subclass passing a "" title to the superclass JTabbedPane through // its insertTab. Later an overridden getTitleAt returns the titles as // managed by the subclass. static class BadPane extends JTabbedPane { private List titles; BadPane() { titles = new ArrayList(1); } @Override public void insertTab( String title, Icon icon, Component component, String tip, int index ) { titles.add(index, title); super.insertTab("", icon, component, tip, index); } @Override public String getTitleAt(int i) { return titles.get(i); } } }