我有以下问题,我正在使用带有HTMLEditorKit的JTextPane,并动态地向窗格添加内容。内容可以是多行,也包含大量图像(小图标)。现在的问题是,如果我通过以下方式插入一堆带图标的文本:
editorKit.insertHTML(doc, doc.getLength(), htmlCode, 0, 0, null);
结果是,html代码中的图像花了很长时间才出现在屏幕上(〜1秒):
问题:有没有办法通过以下方式缓存窗格中显示的图像:
imgsrc = "file:/" + imgSRCOnHDD;
imgsrc = imgsrc.replace("\\", "/");
imgSub = "<img height=\"18\" width=\"18\" style=\"vertical-align:middle;\" src='" + imgsrc + "'></img>";
由于HTMLEditor,我无法使用pane.insertIcon。是否可能有某种容器将其设置为不可见,将内容添加到窗格,然后将容器设置为可见?
答案 0 :(得分:1)
@StanislavL的评论导致我解决了这个问题,虽然不是他提供的链接,它没有工作,我无法找到原因,网址是正确的我仔细检查,但在设置缓存后,窗格总是会显示每个图像的“断链”图片。所以我发现这篇文章,了解HTMLEditorKit的缓存机制:
https://stackoverflow.com/a/27669916/7377320
上述代码:
public class ImageCache extends Hashtable {
public Object get(Object key) {
Object result = super.get(key);
if (result == null){
result = Toolkit.getDefaultToolkit().createImage((URL) key);
put(key, result);
}
return result;
}
}
...
Dictionary cache = (Dictionary) pane.getDocument().getProperty("imageCache");
if (cache == null) {
cache = new ImageCache();
pane.getDocument().putProperty("imageCache", cache);
}
工作完美,现在正在运作。我还对HTMLEditorKit的ImageView进行了此更改:
public class MyImageView extends ImageView {
public MyImageView(Element elem) {
super(elem);
}
@Override
public Icon getNoImageIcon() {
return null;
}
@Override
public Icon getLoadingImageIcon() {
return null;
}
}
这不会在窗格中显示任何损坏的链接图片,也没有“预加载”损坏的链接图片。
一个HTMLEditorKit:
public class MyHTMLEditorKit extends HTMLEditorKit {
@Override
public ViewFactory getViewFactory() {
return new HTMLEditorKit.HTMLFactory() {
public View create(Element e) {
View v = super.create(e);
Object o = e.getAttributes().getAttribute(StyleConstants.NameAttribute);
if (o instanceof HTML.Tag) {
HTML.Tag kind = (HTML.Tag) o;
if (kind == HTML.Tag.IMG) {
return new MyImageView(e);
}
}
if (v instanceof InlineView) {
return new InlineView(e) {
public int getBreakWeight(int axis, float pos, float len) {
return View.GoodBreakWeight;
}
public View breakView(int axis, int p0, float pos, float len) {
if (axis == View.X_AXIS) {
checkPainter();
int p1 = getGlyphPainter().getBoundedPosition(this, p0, pos, len);
if (p0 == getStartOffset() && p1 == getEndOffset()) {
return this;
}
return createFragment(p0, p1);
}
return this;
}
};
} else if (v instanceof ParagraphView) {
return new ParagraphView(e) {
protected javax.swing.SizeRequirements calculateMinorAxisRequirements(int axis, javax.swing.SizeRequirements r) {
if (r == null) {
r = new javax.swing.SizeRequirements();
}
float pref = layoutPool.getPreferredSpan(axis);
float min = layoutPool.getMinimumSpan(axis);
// Don't include insets, Box.getXXXSpan will include them.
r.minimum = (int) min;
r.preferred = Math.max(r.minimum, (int) pref);
r.maximum = Integer.MAX_VALUE;
r.alignment = 0.5f;
return r;
}
};
}
return v;
}
};
}
}