HTML 嵌入 Base64 图像的 JEditorPane 内容类型

问题描述 投票:0回答:2

我正在使用 JeditorPane 和 JEditorKit 来显示一些 HTML。 HTML 显示正确,但图像显示为损坏(它们在浏览器中正确显示)。图片src是base64。我这样设置内容类型:

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

我猜测,因为它同时包含文本和图像,所以内容类型不正确。如果是这样的话,应该设置成什么?短暂性脑缺血发作。

** 马杜山·佩雷拉回复后**

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' 是包含 HTML 的字符串。它是使用 IOUtils.toString 从 html 文件类型读取的。我可能必须开发 SCCE。

实现自定义编辑器

        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);
    }

现在我什么也没得到。显然我没有正确执行你的建议。

java html image swing jeditorpane
2个回答
8
投票

您可以尝试如下所示:

    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 />");

更新:

    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();
    }

更新2:

您必须为

JEditorPane
创建自定义工具包,如下所示:

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;
    }

}

然后你必须重写

getImageURL()
javax.swing.text.html.ImageView
来支持Base64编码的图像:

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);
    }

}

最后您可以将

CustomToolKit
设置为您的
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 />");

0
投票

“Swing 的(默认)HTML 支持不会扩展到 Base 64 编码图像。”,Andrew Thompson 评论, 是

java.net.URL
类受支持协议中的差距。这可以由协议“数据”的附加 URLStreamHandler 填充。

/**
 * Stream handler for protocol "data"
 * @author Sam Ginrich
 *
*/
public class DataURLStreamHandler extends URLStreamHandler
{

    public DataURLStreamHandler()
    {
    }   

    @Override
    protected URLConnection openConnection(final URL u) throws IOException
    {
        return new DataURLConnection(u);
    }

}

/**
 * URLConnection for protocol "data", decoding base64 URLs
 * 
 * @author Sam Ginrich
 *
 */
public class DataURLConnection extends URLConnection
{

    ByteArrayInputStream bis;

    protected DataURLConnection(final URL url)
    {
        super(url);
        bis = null;
    }

    @Override
    public InputStream getInputStream()
    {
        if (bis == null)
        {
            final String path = super.url.getPath(); // URL content, after "data:"
            final String[] parts = path.split(",");
            final String[] encoding = parts[0].split(";");
            if (encoding.length < 2 || !encoding[1].toLowerCase().contains("base64"))
            {
                return null;
            }
            final byte[] chars = Base64.getDecoder().decode(parts[1]);
            bis = new ByteArrayInputStream(chars);
        }
        return bis;
    }

    @Override
    public void connect() throws IOException
    {
    }

}

Java 运行时设计并不意味着集成单个附加协议,因此需要一些反思

/**
 * Factory for DataURLStreamHandler
 * 
 * @author Sam Ginrich
 *
 */
public class DataURLStreamFactory implements URLStreamHandlerFactory
{
    URLStreamHandlerFactory oldFactory;
    Field fFactory;
    Field fHandlerLock;

    /**
     * To be called in a static code segment of a consumer class
     */
    public static void install()
    {
        new DataURLStreamFactory();
    }

    private DataURLStreamFactory()
    {
        try
        {
            // Class URL design does not foresee a set of specific factories,
            // adapt existing instance by reflection
            fFactory = URL.class.getDeclaredField("factory");
            fFactory.setAccessible(true);

            // Protection of URL.factory and URL.handlers
            fHandlerLock = URL.class.getDeclaredField("streamHandlerLock");
            fHandlerLock.setAccessible(true);

            URLStreamHandlerFactory current = (URLStreamHandlerFactory) fFactory.get(null);
            if (current != null && current instanceof DataURLStreamFactory)
            {
                // instance of DataURLStreamFactory already installed
                return;
            }
            oldFactory = current;
            URL.setURLStreamHandlerFactory(this);
        }
        catch (final Exception e)
        {
            fFactory = null;
            e.printStackTrace();
        }
    }

    @Override
    public URLStreamHandler createURLStreamHandler(final String protocol)
    {
        // Option 1, data protocol
        if ("data".equals(protocol))
        {
            return new DataURLStreamHandler();
        }
        else
        {
            URLStreamHandler res = null;
            try
            {
                // Option 2, protocols used by old factory
                if (oldFactory != null)
                {
                    try
                    {
                        res = oldFactory.createURLStreamHandler(protocol);
                    }
                    catch (final Exception mux)
                    {
                        res = null;
                    }
                }

                // Option 3, use built-in factory
                if (res == null)
                {
                    synchronized (fHandlerLock.get(null))
                    {
                        // URL.factory might be other than "this"
                        final URLStreamHandlerFactory current = (URLStreamHandlerFactory) fFactory.get(null);

                        // choose path of built-in factory
                        fFactory.set(null, null);
                        final Method gus = URL.class.getDeclaredMethod("getURLStreamHandler",
                                new Class<?>[] { String.class });
                        gus.setAccessible(true);
                        res = (URLStreamHandler) gus.invoke(null, protocol);
                        fFactory.set(null, current); // re-establish factory
                    }
                }
                return res;
            }
            catch (final Exception e)
            {
                e.printStackTrace();
                return null;
            }
        }
    }
};
© www.soinside.com 2019 - 2024. All rights reserved.