11

I'm trying to create some special component for a specific purpose, on that component I need to draw a HTML string, here's a sample code:

 public class MyComponent extends JComponent{
     public MyComponent(){
        super();
     }

     protected void paintComponent(Graphics g){
        //some drawing operations...
        g.drawString("<html><u>text to render</u></html>",10,10);
     }
 }

Unfortunately the drawString method seems to be not recognizing the HTML format, it foolishly draws the string just as it is.

Is there any way to make that work?

mKorbel
  • 108,320
  • 17
  • 126
  • 296
George Casttrey
  • 407
  • 5
  • 16

7 Answers7

12

If a fan of Java2D; but to get the most leverage from HTML in Swing components and layouts, I'd encourage you to use the component approach suggested by @camickr. If necessary, you can use the flyweight renderer approach seen in JTable, et al, in which a single component is used repeatedly for drawing. The example below is a very simplified outline of the technique, changing only the color and location.

Addendum: Updated example; see also CellRendererPane and Make your apps fly: Implement Flyweight to improve performance.

enter image description here

import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import javax.swing.CellRendererPane;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;

/** @see http://stackoverflow.com/questions/7774960 */
public class PaintComponentTest extends JPanel {

    private static final int N = 8;
    private static final String s = "<html><big><u>Hello</u></html>";
    private JLabel renderer = new JLabel(s);
    private CellRendererPane crp = new CellRendererPane();
    private Dimension dim;

    public PaintComponentTest() {
        this.setBackground(Color.black);
        dim = renderer.getPreferredSize();
        this.add(crp);
    }

    @Override
    protected void paintComponent(Graphics g) {
        super.paintComponent(g);
        for (int i = 0; i < N; i++) {
            renderer.setForeground(Color.getHSBColor((float) i / N, 1, 1));
            crp.paintComponent(g, renderer, this,
                i * dim.width, i * dim.height, dim.width, dim.height);
        }
    }

    private void display() {
        JFrame f = new JFrame("PaintComponentTest");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(this);
        f.pack();
        f.setSize(dim.width * N, dim.height * (N + 1));
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new PaintComponentTest().display();
            }
        });
    }
}
trashgod
  • 196,350
  • 25
  • 213
  • 918
  • thank you very much for the help. I've found out another way to do this without needing the CellRendererPane. (Take a look on my answer post). – George Casttrey Oct 15 '11 at 14:27
  • 1
    You're welcome. [`CellRendererPane`](http://download.oracle.com/javase/7/docs/api/javax/swing/CellRendererPane.html) offers a few convenient optimizations for buffering and validation that may yet prove useful going forward. – trashgod Oct 15 '11 at 15:12
10

As others have commented, Swing components support HTML 3.2 and basic styles.

For details on how to leverage that ability in the paintComponent(Graphics) method, see the LabelRenderTest.java source on this thread.

The approach is to render the label to an image, then render the image to the Graphics object.

Community
  • 1
  • 1
Andrew Thompson
  • 163,965
  • 36
  • 203
  • 405
  • I've found out a way to paint the htmlString using JLabel but without the need to render an image (take a look on my post). Thank you very much for the help. – George Casttrey Oct 15 '11 at 14:24
6

The answer to your question is that HTML is not supported.

The easy solution is to use Swing components and lay them out properly.

If you want to do it yourself then you need to manipulate the Font that you use for the drawString() method. You can use a Font with italics, bold etc.

camickr
  • 308,339
  • 18
  • 152
  • 272
6

I've found a short and a clean way to simulate the paintHtmlString; here's the code:

public class MyComponent extends JComponent {

    private JLabel label = null;

    public MyComponent() {
        super();
    }

    private JLabel getLabel() {
        if (label == null) {
            label = new JLabel();
        }
        return label;
    }

    /**
     * x and y stand for the upper left corner of the label
     * and not for the baseline coordinates ...
     */
    private void paintHtmlString(Graphics g, String html, int x, int y) {
        g.translate(x, y);
        getLabel().setText(html);
        //the fontMetrics stringWidth and height can be replaced by
        //getLabel().getPreferredSize() if needed
        getLabel().paint(g);
        g.translate(-x, -y);
    }

    protected void paintComponent(Graphics g) {
        //some drawing operations...
        paintHtmlString(g, "<html><u>some text</u></html>", 10, 10);
    }
}

Thanks to every one for the help, I do appreciate that very much.

trashgod
  • 196,350
  • 25
  • 213
  • 918
George Casttrey
  • 407
  • 5
  • 16
  • As noted [here](https://stackoverflow.com/a/5853992/1143274), it may be necessary to `getLabel().setSize(getLabel().getPreferredSize())` if the JLabel has never been added under a Frame that has been `pack()`ed (or equivalent). – Evgeni Sergeev Oct 09 '18 at 19:43
5

This is not the way to implement.

The drawString(String str, int x, int y) only draws the text given by the specified string, using this graphics context's current font and color.

You can get more information for your problem from here, How to Use HTML in Swing Components

COD3BOY
  • 11,484
  • 1
  • 34
  • 55
4

Use JLabel as your base class. Graphics.drawString() can only... draw a string.

Michał Šrajer
  • 27,013
  • 6
  • 52
  • 75
  • My component doesn't just draw a html string, there's plenty of other drawing operations which i didn't want to show in the sample code just to focus on the real problem. I know that JLabel and all other Swing components which are holding text do support the html format.
    Im just wondering if there's any way to render html texts through a Graphics instance.
    – George Casttrey Oct 15 '11 at 01:02
  • Thank you for the helpful answer. – George Casttrey Oct 15 '11 at 14:30
2

There isn't an easy way to render HTML to paint with Graphics, go with a JLabel.

However, there is an interesting way to accomplish this by creating a JLabel with HTML and painting it's graphics to the component:

private JLabel label;

public MyComponent() {
    label = new JLabel("Before Red");
    label.setText("<html><u>test</u></html>");
    this.add(label);
}

public void repaint(Graphics g){
    g = label.getGraphics();
}

The easiest way to accomplish this would be to set a font for your graphics, for example:

public void repaint(Graphics g){    
    Font f = new Font("Courier", Font.BOLD, 12);
    g.setFont(f);
    g.drawString("Bolded Courier", 5, 15);
}
jpalm
  • 1,997
  • 1
  • 19
  • 24