问题描述
我正在使用JeditorPane和JEditorKit来显示一些HTML。 HTML正确显示但图像显示为已损坏(它们在浏览器中正确显示)。图像src是base64。我这样设置内容类型:
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");
我猜这是因为它包含文字和图片,内容类型不正确。如果是这种情况,应该设置什么? TIA。
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.
** Madushan Perera回复后**
** 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'是包含HTML的字符串。它是使用IOUtils.toString从html文件类型中读取的。我可能需要开发一个SCCE。
'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.
实现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.
推荐答案
你可以尝试下面这样的事情:
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 />");
更新:
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
创建自定义工具包,如下所示:
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;
}
}
然后你必须覆盖 getImageURL()
javax.swing.text.html.ImageView
支持Base64编码图像。
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);
}
}
最后你可以设置你的 CustomTollKit
到您的 editorPane
:
Finally you can set your CustomTollKit
to your editorPane
:
String imgURL = ".........";
editor_pane.setContentType("text/html");
editor_pane.setEditable(false);
CustomToolKit tooKit = new CustomToolKit();
editor_pane.setEditorKit(tooKit);
editor_pane.setText("<h1>Image Title</h1><img src='" + imgURL + "' alt='img_alt' width='150' height='150' /><br />");
这篇关于用于HTML嵌入式Base64图像的JEditorPane内容类型的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!