javahtmlimageswingjeditorpane

JEditorPane Content Type for HTML Embedded Base64 Images


I am using JeditorPane and JEditorKit to display some HTML. The HTML is displaying correctly but the images show up as broken (they display properly in a browser). The image src is base64. I set the content type thus:

final JEditorPane ed=new JEditorPane();
ed.setContentType("text/html");

I'm guessing that because it contains both text and images, the content type is incorrect. If that is the case, what should it be set to? TIA.

** After Madushan Perera's reply**

final JEditorPane ed=new JEditorPane();
ed.setContentType("text/html");
ed.setEditable(false);
HTMLDocument html=(HTMLDocument) ed.getDocument();
html.putProperty("IgnoreCharsetDirective", new Boolean(true));
HTMLEditorKit htmle=(HTMLEditorKit) ed.getEditorKit();
try {
    htmle.insertHTML(html,html.getLength(),content,0,0,null);
} catch (BadLocationException | IOException e) {
    // Should not get here
    e.printStackTrace();
}
ed.addHyperlinkListener(new HyperlinkListener() {
          public void hyperlinkUpdate(final HyperlinkEvent pE) {
              if (HyperlinkEvent.EventType.ACTIVATED == pE.getEventType()) {
                  String desc = pE.getDescription();
                  if (desc == null || !desc.startsWith("#")) return;
                  desc = desc.substring(1);
                  ed.scrollToReference(desc);
              }
          }
      });
ed.setCaretPosition(0);
JScrollPane scroll=new JScrollPane(ed,JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED,JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
JPanel jp=new JPanel();
Dimension size=new Dimension(700,700);
jp.setPreferredSize(size);
jp.setLayout(new BorderLayout());
jp.add(scroll);
        JOptionPane.showMessageDialog(null,jp,title,JOptionPane.INFORMATION_MESSAGE);

'html' is the string containing the HTML. It was read from an html file type using IOUtils.toString. I'm probably going to have to develop an SCCE.

Implementing CustomEditor

        final JEditorPane ed=new JEditorPane();
        ed.setContentType("text/html");
        ed.setEditable(false);
        CustomToolKit htmle=new CustomToolKit();
        ed.setEditorKit(htmle);
        String content=readFile(fileName_+".html").replaceAll("(\\r|\\n)", "");
        content=content.replace("!!!!",VERSION.VERSION);
        ed.setText(content);
        ed.addHyperlinkListener(new HyperlinkListener() {
            public void hyperlinkUpdate(final HyperlinkEvent pE) {
                if (HyperlinkEvent.EventType.ACTIVATED == pE.getEventType()) {
                    String desc = pE.getDescription();
                    if (desc == null || !desc.startsWith("#")) return;
                    desc = desc.substring(1);
                    ed.scrollToReference(desc);
                }
            }
        });
        ed.setCaretPosition(0);
        JScrollPane scroll=new JScrollPane(ed,JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED,JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
        JPanel jp=new JPanel();
        Dimension size=new Dimension(700,700);
        jp.setPreferredSize(size);
        jp.setLayout(new BorderLayout());
        jp.add(scroll);
        JOptionPane.showMessageDialog(null,jp,title,JOptionPane.INFORMATION_MESSAGE);
    }

Now I get nothing. Obviously I did not properly implement your suggestion.


Solution

  • You can try something like below :

        String imgsrc = this.getClass().getClassLoader()
                .getResource("your_package_name/image.png").toString();
    
        editor_pane.setContentType("text/html");
        editor_pane.setEditable(false);
        editor_pane.setText("<h3>Image Title</h3><img src='" + imgsrc + "' alt='img' name='img_name' width='100' height='100' /><br />");
    

    UPDATE :

        String imgURL = "";
    
        String encodedImg = imgURL.split(",")[1];
        byte[] decodedImg = Base64.getDecoder().decode(encodedImg.getBytes(StandardCharsets.UTF_8));
    
        Path destinationFile = Paths.get("C:\\your_path\\", "myImage.jpg");
        try {
            Path path = Files.write(destinationFile, decodedImg);
            editor_pane.setContentType("text/html");
            editor_pane.setEditable(false);
            editor_pane.setText("<h3>Image Title</h3><img src='" + path.toAbsolutePath().toUri() + "' alt='img_alt' width='150' height='150' /><br />");
        } catch (IOException ex) {
            ex.printStackTrace();
        }
    

    UPDATE 2 :

    You have to create custom toolkit for the JEditorPane like below :

    public class CustomToolKit extends HTMLEditorKit {
    
        private static HTMLFactory factory = null;
    
        @Override
        public ViewFactory getViewFactory() {
            if (factory == null) {
                factory = new HTMLFactory() {
    
                    @Override
                    public View create(Element elem) {
                        AttributeSet attrs = elem.getAttributes();
                        Object elementName = attrs.getAttribute(AbstractDocument.ElementNameAttribute);
                        Object o = (elementName != null) ? null : attrs.getAttribute(StyleConstants.NameAttribute);
                        if (o instanceof HTML.Tag) {
                            HTML.Tag kind = (HTML.Tag) o;
                            if (kind == HTML.Tag.IMG) {
                                return new BASE64ImageView(elem);
                            }
                        }
                        return super.create(elem);
                    }
                };
            }
            return factory;
        }
    
    }
    

    Then you have to override the getImageURL() of javax.swing.text.html.ImageView to support Base64 encoded images:

    public class BASE64ImageView extends ImageView {
    
        private URL url;
    
        public BASE64ImageView(Element elmnt) {
            super(elmnt);
            populateImage();
        }
    
        private void populateImage() {
            Dictionary<URL, Image> cache = (Dictionary<URL, Image>) getDocument()
                    .getProperty("imageCache");
            if (cache == null) {
                cache = new Hashtable<>();
                getDocument().putProperty("imageCache", cache);
            }
    
            URL src = getImageURL();
            cache.put(src, loadImage());
    
        }
    
        private Image loadImage() {
            String b64 = getBASE64Image();
            BufferedImage newImage = null;
            ByteArrayInputStream bais = null;
            try {
                bais = new ByteArrayInputStream(
                        Base64.getDecoder().decode(b64.getBytes()));
                newImage = ImageIO.read(bais);
            } catch (Throwable ex) {
                ex.printStackTrace();
            }
            return newImage;
        }
    
        @Override
        public URL getImageURL() {
            String src = (String) getElement().getAttributes()
                    .getAttribute(HTML.Attribute.SRC);
            if (isBase64Encoded(src)) {
    
                this.url = BASE64ImageView.class.getProtectionDomain()
                        .getCodeSource().getLocation();
    
                return this.url;
            }
            return super.getImageURL();
        }
    
        private boolean isBase64Encoded(String src) {
            return src != null && src.contains("base64,");
        }
    
        private String getBASE64Image() {
            String src = (String) getElement().getAttributes()
                    .getAttribute(HTML.Attribute.SRC);
            if (!isBase64Encoded(src)) {
                return null;
            }
            return src.substring(src.indexOf("base64,") + 7, src.length() - 1);
        }
    
    }
    

    Finally you can set your CustomToolKit to your editorPane :

            String imgURL = ".........";
            editor_pane.setContentType("text/html");
            editor_pane.setEditable(false);
            CustomToolKit toolKit = new CustomToolKit();
            editor_pane.setEditorKit(toolKit);
            editor_pane.setText("<h1>Image Title</h1><img src='" + imgURL + "' alt='img_alt' width='150' height='150' /><br />");