- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我在图表应用程序中显示我的 SVG 绘图时遇到问题。 Canvas 采用其父组件的大小,而不是它包含的文档的大小。大于此大小的文档不会完全呈现。
案例
我有一幅画,例如,621x621 像素。
<?xml version="1.0" encoding="UTF-8"?>
<svg xmlns:xlink="http://www.w3.org/1999/xlink" xmlns="http://www.w3.org/2000/svg" contentScriptType="text/ecmascript" zoomAndPan="magnify" contentStyleType="text/css" preserveAspectRatio="xMaxYMax slice" version="1.0">
<rect x="50" y="50" fill="white" width="20" height="20" stroke="black" stroke-width="1"/>
<rect x="600" y="600" fill="blue" width="20" height="20" stroke="black" stroke-width="1"/>
</svg>
它没有 viewBox,因为它是动态创建的。当我创建文档时,我没有尺寸。
我在 JSVGScrollPane
中有一个 JSVGCanvas
,滚动 Pane 的父级比文档小,比如说它的 500x500 像素。
预期行为
当您启动应用程序时,适合 500x500 的绘图部分是可见的。底层文档更大,所以有滚动条允许我向左和向下滚动 121 像素。
当你增加框架的大小时,越来越多的绘图变得可见。当您将其增加到大于 620x620px 时,滚动条可能会完全消失。
如果再次使面板小于文档,滚动条会重新出现。
遇到的行为
当您启动应用程序时,适合 500x500 的绘图部分是可见的。没有滚动条可见。
当您使面板变大时, Canvas 会增加(其背景颜色)但绘图不会扩展到初始 500x500px 之外的部分。
当您将面板缩小到 400x400 时,确实会出现滚动条,但只允许我滚动 100 像素。所以 Canvas 仍然是初始的 500x500 而不是所需的 621x621。
示例
可以使用 ScrollExample
重现遇到的行为在 Batik 存储库和我在上面添加的 SVG 文件上。该示例使用 500x500 像素的固定大小。
如果您要为 Canvas 和滚动 Pane 添加背景颜色,则整个框架都是蓝色的。因此,虽然 Canvas 确实增加了尺寸,但它不会绘制超出其初始尺寸的任何东西。
canvas.setBackground(Color.blue);
scroller.setBackground(Color.red);
我可以在调整面板大小时重置文档。这使得它重新绘制绘图的可见部分,但滚动仍然没有按预期工作,因为 Canvas 仍然只是可见部分的大小而不是文档的大小。您可以通过将此监听器添加到框架(或我想的任何其他面板)来对此进行测试。
class ResizeListener implements ComponentListener {
JSVGCanvas canvas;
URI url;
public ResizeListener(JSVGCanvas c, URI u) {
canvas = c;
url = u;
}
@Override
public void componentResized(ComponentEvent arg0) {
canvas.setURI(url.toString());
// Calling invalidate on canvas or scroller does not work
}
// ...
};
我读到过我应该有一个 viewBox,但是我的文档经常被重新创建(基于 modelchange-event)并且当我创建文档时,我不知道如何获取 viewBox。 SVGDocument.getRootElement().getBBox();
通常给我 null
。当没有给出 viewBox 时,Batik 代码中应该有回退,所以我希望这是可选的。此外,当我在某处平移时,如果我更改文档,它应该保持平移。
我希望我能把我的问题说清楚。当我期望 JSVG 类提供开箱即用的我想要的行为时,我是否期望太多,或者我在这里遗漏了什么?有人可以指导我找到解决方案吗?
最佳答案
我想我可能找到了解决方案。我需要设置文档的高度和宽度,而不是弄乱 viewBox。设置大小后,不需要 viewBox 并且滚动条按预期工作。如果我确实设置了一个 View 框, Canvas 会不断缩放图像以使其适合。
现在,为了更新大小,每当我更改可能影响大小的 dom 时,我都会使用它。
static BridgeContext ctx = new BridgeContext(new UserAgentAdapter());
static GVTBuilder builder = new GVTBuilder();
private static void calculateSize(SVGDocument doc) {
GraphicsNode gvtRoot = builder.build(ctx, doc);
Rectangle2D rect = gvtRoot.getSensitiveBounds();
doc.getRootElement().setAttributeNS(null,
SVGConstants.SVG_WIDTH_ATTRIBUTE, rect.getMaxX() + "");
doc.getRootElement().setAttributeNS(null,
SVGConstants.SVG_HEIGHT_ATTRIBUTE, rect.getMaxY() + "");
}
我在 UpdateManager
线程的之外 执行此操作,在修改 dom 之后,我使用 JSVGCanvas.setSVGDocument()
。当我尝试通过 UpdateManager
执行此操作时,它只更新了第一个更改。正确初始化 updatequeue 可能会解决此问题,但由于我更改了很多文档,所以我每次都可以重新开始。
我还有一个小问题。每当我设置新文档时,滚动位置都会重置。我尝试在操作之前进行转换并在之后重新应用它,但这似乎不起作用。
编辑+:我设法通过替换文档的根节点而不是替换整个文档来解决这个小问题。 Canvas 设置为文档状态 ALWAYS_DYNAMIC。文档中的更改会立即应用,包括根节点的新大小(如上所述设置)。但是因为文档没有完全被替换,所以滚动状态被保留。
Canvas 仅在完成渲染文档后才准备好修改 dom。我使用 SVGLoad-listener 来检测这一点。
class Canvas extends JSVGCanvas implements SVGLoadEventDispatcherListener {
public Canvas() {
super();
setDocumentState(ALWAYS_DYNAMIC);
addSVGLoadEventDispatcherListener(this);
}
/**
* Indicatates whether the canvas has finished its first render, the
* canvas is now ready for modification of the dom
*/
private boolean isReadyForModification = false;
/**
* Renew the document by replacing the root node with the one of the new
* document
*
* @param doc The new document
*/
public void renewDocument(final SVGDocument doc) {
if (isReadyForModification) {
getUpdateManager().getUpdateRunnableQueue().invokeLater(
new Runnable() {
@Override
public void run() {
// Get the root tags of the documents
Node oldRoot = getSVGDocument().getFirstChild();
Node newRoot = doc.getFirstChild();
// Make the new node suitable for the old
// document
newRoot = getSVGDocument().importNode(newRoot,
true);
// Replace the nodes
getSVGDocument().replaceChild(newRoot, oldRoot);
}
});
} else {
setSVGDocument(doc);
}
}
@Override
public void svgLoadEventDispatchCompleted(SVGLoadEventDispatcherEvent e) {
isReadyForModification = true;
}
// ...
}
关于java - 使 JSVGScrollPane 内的 JSVGCanvas 与 SVGDocument 大小匹配,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10838971/
我正在尝试使用 Batik 在 JSVGCanvas 中显示 SVG 文件。在这个 SVG 文件中有一个图像。问题是图像没有显示。相反,我得到了一个错误图像。这是我的代码: SVG 文件 JA
使用:Apache batik (SVG)、NetBeans IDE、NetBeans 平台、Java 我有一个使用 batik 和 JSVGCanvas 组件显示 SVG 的应用程序。 我还调用了
我似乎在使用 batikSVG 使用 Java 操作 SVG 时遇到问题。我可以在 JSVG Canvas 上很好地显示 SVG,但是当我尝试使用 getSVGDocument 访问 Canvas 的
我一直在遵循本指南 http://softwareisart.blogspot.co.uk/2011/11/drag-and-drop-of-complex-custom-objects.html 使
我有一个来自 java 的 Batik 库的 JSVGCanvas 对象。在我的应用程序中,我在原理图中渲染了多个对象。我需要知道鼠标下方的组件,以便我可以渲染从外部源渲染的适当的工具提示和描述。 我
浏览 documentation for JSVGCanvas ,看来我可能无法做到这一点。但是,这很有意义。如何从 String 变量而不是 File 创建 JSVGCanvas? 我的想法是我可以
我的程序似乎工作得很好,但当我尝试调整组件大小时,我时不时地收到“IllegalStateExceptions:RunnableQueue 未启动或已退出”。我已将 documentState 设置为
我在图表应用程序中显示我的 SVG 绘图时遇到问题。 Canvas 采用其父组件的大小,而不是它包含的文档的大小。大于此大小的文档不会完全呈现。 案例 我有一幅画,例如,621x621 像素。
我花了 7 个小时尝试让 Batik 中的 JSVGCanvas 显示一个简单的 svg 文件。我无法让它发挥作用。有趣的是,鉴于我正在使用的依赖项的版本,我总是收到不同的错误消息。我上次的尝试给了我
我是一名优秀的程序员,十分优秀!