Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Why does the JTable header not appear in the image?

I was offering advice on capturing an image of tabular data on Java API or Tool to convert tabular data into PNG image file - when the OP requested a code sample. Turns out to be harder than I thought! The JTable header vanishes from the PNG that the code writes.

PNG

PNG

Screen shot

enter image description here

import javax.swing.*; import java.awt.Graphics; import java.awt.BorderLayout; import java.awt.image.BufferedImage;  import javax.imageio.ImageIO; import java.io.File;  class TableImage {      public static void main(String[] args) throws Exception {         Object[][] data = {             {"Hari", new Integer(23), new Double(78.23), new Boolean(true)},             {"James", new Integer(23), new Double(47.64), new Boolean(false)},             {"Sally", new Integer(22), new Double(84.81), new Boolean(true)}         };          String[] columns = {"Name", "Age", "GPA", "Pass"};          JTable table = new JTable(data, columns);         JScrollPane scroll = new JScrollPane(table);         JPanel p = new JPanel(new BorderLayout());         p.add(scroll,BorderLayout.CENTER);          JOptionPane.showMessageDialog(null, p);          BufferedImage bi = new BufferedImage(             (int)p.getSize().getWidth(),             (int)p.getSize().getHeight(),             BufferedImage.TYPE_INT_RGB             );          Graphics g = bi.createGraphics();         p.paint(g);          JOptionPane.showMessageDialog(null, new JLabel(new ImageIcon(bi)));         ImageIO.write(bi,"png",new File("table.png"));     } } 

Note: I checked over camickr's Screen Image class and included a call to the doLayout(Component) method. The method is useful for if a Component has never been realized on screen, but has no effect on this code (which pops the panel containing the table in an option pane before trying to render it).

What is needed in order to get the table header to render?

Update 1

Changing the line..

        p.paint(g); 

..to (with an appropriate import)..

        p.paint(g);         JTableHeader h = table.getTableHeader();         h.paint(g); 

..produces..

2nd try

I'll keep tweaking it.

Update 2

kleopatra (strategy 1) & camickr (strategy 2) have provided an answer each, both of which work, & neither of which requires adding the JTable to a dummy component (which is an huge hack IMO).

While strategy 2 will crop (or expand) to 'just the table', the 1st strategy will capture the panel containing the table. This becomes problematic if the table contains many entries, showing an image of a truncated table with a scroll bar.

While strategy 1 might be further tweaked to get around that, I really like the neat simplicity of strategy 2, so it gets the tick.

As pointed out by kleopatra, there was no 'tweak' needed. So I'll try again..

Update 3

This is the image produced by the methods put forward by both camickr and kleopatra. I'd have put it twice, but to my eye, they are identical (though I have not done a pixel by pixel comparison).

JTable in Nimbus PLAF

import javax.swing.*; import java.awt.*; import java.awt.image.BufferedImage;  import javax.imageio.ImageIO; import java.io.File;  class TableImage {      String[] columns = {"Name", "Age", "GPA", "Pass"};     /** Any resemblance to persons living or dead is purely incidental. */     Object[][] data = {         {"André", new Integer(23), new Double(47.64), new Boolean(false)},         {"Jeanie", new Integer(23), new Double(84.81), new Boolean(true)},         {"Roberto", new Integer(22), new Double(78.23), new Boolean(true)}     };      TableImage() {     }      public JTable getTable() {         JTable table = new JTable(data, columns);         table.setGridColor(new Color(115,52,158));         table.setRowMargin(5);         table.setShowGrid(true);          return table;     }      /** Method courtesy of camickr.     https://stackoverflow.com/questions/7369814/why-does-the-jtable-header-not-appear-in-the-image/7375655#7375655     Requires ScreenImage class available from..     http://tips4java.wordpress.com/2008/10/13/screen-image/ */     public BufferedImage getImage1(JTable table) {         JScrollPane scroll = new JScrollPane(table);          scroll.setColumnHeaderView(table.getTableHeader());         table.setPreferredScrollableViewportSize(table.getPreferredSize());          JPanel p = new JPanel(new BorderLayout());         p.add(scroll, BorderLayout.CENTER);          BufferedImage bi = ScreenImage.createImage(p);         return bi;     }      /** Method courtesy of kleopatra.     https://stackoverflow.com/questions/7369814/why-does-the-jtable-header-not-appear-in-the-image/7372045#7372045 */     public BufferedImage getImage2(JTable table) {         JScrollPane scroll = new JScrollPane(table);          table.setPreferredScrollableViewportSize(table.getPreferredSize());          JPanel p = new JPanel(new BorderLayout());         p.add(scroll, BorderLayout.CENTER);          // without having been shown, fake a all-ready         p.addNotify();          // manually size to pref         p.setSize(p.getPreferredSize());          // validate to force recursive doLayout of children         p.validate();          BufferedImage bi = new BufferedImage(p.getWidth(), p.getHeight(), BufferedImage.TYPE_INT_RGB);          Graphics g = bi.createGraphics();         p.paint(g);         g.dispose();          return bi;     }      public void writeImage(BufferedImage image, String name) throws Exception {         ImageIO.write(image,"png",new File(name + ".png"));     }      public static void main(String[] args) throws Exception {         UIManager.setLookAndFeel("com.sun.java.swing.plaf.nimbus.NimbusLookAndFeel");         TableImage ti = new TableImage();         JTable table;         BufferedImage bi;          table = ti.getTable();         bi = ti.getImage1(table);         ti.writeImage(bi, "1");         JOptionPane.showMessageDialog(null, new JLabel(new ImageIcon(bi)));          table = ti.getTable();         bi = ti.getImage2(table);         ti.writeImage(bi, "2");         JOptionPane.showMessageDialog(null, new JLabel(new ImageIcon(bi)));     } } 

Both achieve the goal. Using camickr's method you leverage the further power of the ScreenImage API. Using kleopatra's method - about a dozen lines (less the comments and white space) of pure J2SE.

While ScreenImage is a class I will use and recommend in future, the other approach using core J2SE is what I'd probably use for this exact circumstance.

So while the 'tick' will stay with camickr, the bounty is going to kleopatra.

like image 885
Andrew Thompson Avatar asked Sep 10 '11 05:09

Andrew Thompson


People also ask

What is JTable and how it is created?

The JTable class is a part of Java Swing Package and is generally used to display or edit two-dimensional data that is having both rows and columns. It is similar to a spreadsheet. This arranges data in a tabular form. Constructors in JTable: JTable(): A table is created with empty cells.

How do I search for a JTable?

We can implement the search functionality of a JTable by input a string in the JTextField, it can search for a string available in a JTable. If the string matches it can only display the corresponding value in a JTable. We can use the DocumentListener interface of a JTextField to implement it.

Which one is an example of constructor of JTable?

There are two JTable constructors that directly accept data ( SimpleTableDemo uses the first): JTable(Object[][] rowData, Object[] columnNames) JTable(Vector rowData, Vector columnNames)


2 Answers

that was a tough one

Was puzzled that that header didn't show up on the image at all: it wasn't clipped or something, it wasn't painted at all. The reason for that is .. that at the time of painting the panel to the image, the header is no longer part of the hierarchy. When closing the optionPane, table.removeNotify removes the header. For adding it again, call addNotify, as in this snippet:

    JScrollPane scroll = new JScrollPane(table);     JPanel p = new JPanel(new BorderLayout());     p.add(scroll, BorderLayout.CENTER);     JOptionPane.showMessageDialog(null, p);     table.addNotify();     p.doLayout();     BufferedImage bi = new BufferedImage(p.getWidth() + 100,             p.getHeight() + 100, BufferedImage.TYPE_INT_RGB);      Graphics g = bi.createGraphics();     p.paint(g);     g.dispose(); 

[solved in edit] What I still don't understand why the panel shows up empty without first having been shown in the optionPane - typically some combination of ..

   p.doLayout();    p.setSize(p.getPreferredSize())  

... will do

Edit

last confusion solved: to force a recursive re-layout of the pane, all container along the line must be driven into believing they have a peer - validate is optimized to do nothing if not. Doing the addNotify on the panel (instead of on the table) plus validate does the trick

    //        JOptionPane.showMessageDialog(null, p);     // without having been shown, fake a all-ready     p.addNotify();     // manually size to pref     p.setSize(p.getPreferredSize());     // validate to force recursive doLayout of children     p.validate();     BufferedImage bi = new BufferedImage(p.getWidth() + 100,             p.getHeight() + 100, BufferedImage.TYPE_INT_RGB);      Graphics g = bi.createGraphics();     p.paint(g);     g.dispose();      JOptionPane.showMessageDialog(null, new JLabel(new ImageIcon(bi))); 

Not tested for side-effects.

Edit 2

Just grumbling a bit about the..

strategy 1 might be further tweaked to get around that [truncated table column]

Actually there is no tweak needed (or the the same tweak needed as for ScreenImage, depends on what you consider a tweak :) - both require to make the JScrollPane not do its job by setting the table's prefViewportSize, the exact same line in both:

    // strategy 1     table.setPreferredScrollableViewportSize(table.getPreferredSize());     p.addNotify();      // strategy 2     scroll.setColumnHeaderView(table.getTableHeader());     table.setPreferredScrollableViewportSize(table.getPreferredSize()); 
like image 124
kleopatra Avatar answered Oct 14 '22 22:10

kleopatra


It was not a requirement of this thread to render the table without first displaying it, but that was the ultimate goal

ScreenImage handles this.

You must manually add the header to the scrollpane.

import javax.swing.*; import java.awt.Graphics; import java.awt.BorderLayout; import java.awt.image.BufferedImage;  import javax.imageio.ImageIO; import java.io.File;  class TableImage {      public static void main(String[] args) throws Exception {         Object[][] data = {             {"Hari", new Integer(23), new Double(78.23), new Boolean(true)},             {"James", new Integer(23), new Double(47.64), new Boolean(false)},             {"Sally", new Integer(22), new Double(84.81), new Boolean(true)}         };          String[] columns = {"Name", "Age", "GPA", "Pass"};          JTable table = new JTable(data, columns);         JScrollPane scroll = new JScrollPane(table);          scroll.setColumnHeaderView(table.getTableHeader());         table.setPreferredScrollableViewportSize(table.getPreferredSize());          JPanel p = new JPanel(new BorderLayout());         p.add(scroll, BorderLayout.CENTER);          BufferedImage bi = ScreenImage.createImage(p);          JOptionPane.showMessageDialog(null, new JLabel(new ImageIcon(bi)));         ImageIO.write(bi,"png",new File("table.png"));     } } 

Note: Kleopatra's suggeston to use the addNotify() on the panel will not work with ScreenImage. The addNotify() method makes the component displayable and the ScreenImage code will only lay out the components for non-displayable components. I might look into making this more general.

like image 23
camickr Avatar answered Oct 14 '22 23:10

camickr