通过javax.swing.text.Element从JTextPane获取组件?

通过javax.swing.text.Element从JTextPane获取组件?,java,swing,element,jtextpane,jcomponent,Java,Swing,Element,Jtextpane,Jcomponent,我使用JTextPane来显示字符和符号,其中后者由自定义绘制的JComponents表示。例如,文本窗格可能显示如下内容: 文本窗格是用户可编辑的,允许用户通过按钮在任何位置添加更多符号,并替换选定的文本。我通过JTextPane.insertComponent()方法来实现这一点。在应用程序中的某个时刻,我需要知道文本窗格中当前显示的内容,我指的不仅是输入的文本,还包括其中包含的确切组件 为了管理文本窗格的内容,我在位置和文档监听器方面遇到了很多麻烦,但我一直在制造比我解决的问题更多的问题。

我使用
JTextPane
来显示字符和符号,其中后者由自定义绘制的
JComponents
表示。例如,文本窗格可能显示如下内容: 文本窗格是用户可编辑的,允许用户通过按钮在任何位置添加更多符号,并替换选定的文本。我通过
JTextPane.insertComponent()方法来实现这一点。在应用程序中的某个时刻,我需要知道文本窗格中当前显示的内容,我指的不仅是输入的文本,还包括其中包含的确切组件

为了管理文本窗格的内容,我在
位置
文档监听器
方面遇到了很多麻烦,但我一直在制造比我解决的问题更多的问题。这就是为什么我最终决定,我的问题可能是由于我的设计错误,所以我决定看看,如果我不能通过文本窗格访问我的组件

通过搜索
AbstractDocument
和其他相关类的文档和源代码,我找到了接口
javax.swing.text.Element
。然后让我的应用程序输出

for(int i = 0; i < textPane.getDocument().getLength(); i++) {
    System.out.println(((StyledDocument) textPane.getDocument()).getCharacterElement(i));
}
for(int i=0;i
这给了我:

叶素(含量)0,4

叶素(含量)0,4

叶素(含量)0,4

叶素(含量)0,4

叶素(组分)4,5

叶素(含量)5,9

叶素(含量)5,9

叶素(含量)5,9

叶素(含量)5,9

叶素(组分)9,10

看到我得到的
leaveElements
似乎确实有一些关于
文档中哪个位置显示的信息,我想一定有可能在该位置获得实际内容。在又搜索了半个小时如何获取每个元素所代表的内容后,我放弃了,决定在这里发布我的问题,希望你们中的一些人可能知道如何实现这一点

我看到有人试图通过
textPane.getComponents()
访问组件,它返回一个组件数组,其中包含
JTextPane
中实际包含的组件的确切数量,但它们都是
javax.swing.text.ComponentView$invalidater
类型,这显然对我没用。也许我只是不明白如何从这里继续下去,因为对我的符号的原始类型的转换不起作用

tl;dr


如何获取
JTextPane
文本中的
JComponent
,以及它在文本窗格中的位置?

您可以遍历文本窗格的
StyledDocument
以查找表示组件或图标的元素,如下所示

BranchElement(section) 0,7 BranchElement(paragraph) 0,7 LeafElement(content) 0,4 LeafElement(icon) 4,5 class javax.swing.plaf.IconUIResource LeafElement(component) 5,6 class javax.swing.JLabel LeafElement(content) 6,7


原始组件是
javax.swing.text.ComponentView$invalidater的第一个(也是唯一的)子组件,您可以从ComponentView中看到这一点


您可以获得无效器列表,并使用其子级访问插入的组件。

希望这有助于提供有关定位的一些想法+1更多关注:-)@Gagandeep-Bali感谢您的关注:-)。您的答案很有趣,但不幸的是,它不适用于我的问题,因为我不需要知道组件的
位置
,而是需要知道文本中的位置:index。但是,如果我无法访问组件本身,那么所有这些都是无用的,这对我的目标来说是绝对必要的。啊哈,到目前为止,我还无法掌握这些知识。祝有智慧的人,能够很快就把话题讲清楚:-)祝你好运。。。这篇文章可能会有帮助。@extremecorders确实有帮助。这样我就可以正确地访问我的所有组件。我还不能完全接受这个答案,原因有二:1。你没有把它作为一个答案和2。我无法通过这种方式获得每个组件的位置,或者可以吗?(说到位置,我指的不是位置,而是索引。)参见ComponentView的源代码。实际上,在布局视图上,调用了操作setComponentParent。它的代码为void setComponentParent(){View p=getParent();if(p!=null){Container parent=getContainer();if(parent!=null){if(c==null){//尝试构建组件comp=createComponent();if(comp!=null){createdC=comp;c=new invalidater(comp);}现在,这就是我一直在寻找的答案!我知道必须能够通过
元素
访问
JTextPane
中的所有内容,我就是不知道如何访问。后续问题:1.如何发现我必须向
属性集
查询
样式常量。ComponentElementName
,我是说n我如何知道在何处查找此类字符串常量?2.为什么
element迭代器
在文档内容后给我一个新行字符?(我得到
leaveelement(content)12,13
,即使
textPane.getDocument.getLength()
产生
12
!)@SheridanVespo:1.老实说,我枚举了
getAttributeNames()
,并在API中搜索了命名常量。2.感谢
AbstractDocument#toString()
。1.谢谢,也许这会帮助我解决未来的问题。2.
AbstractDocument
没有
toString()
method!?至少不是根据。@SheridanVespo:Ah,
AbstractDocument.BranchElement#toString()
;我在
元素.toString()
中放置了一个断点。但是该函数返回“BranchElement(“+getName()+”)+getStartOffset()+”,“+getEndOffset()+”\n”
所以,这并不能解释
/**
 * @see http://stackoverflow.com/a/15669307/230513
 * @see http://stackoverflow.com/questions/2883413
 */
public class DocumentParse {

    private static final String ELEM = AbstractDocument.ElementNameAttribute;
    private static final String ICON = StyleConstants.IconElementName;
    private static final String COMP = StyleConstants.ComponentElementName;

    public static void main(String args[]) throws Exception {
        JFrame f = new JFrame();
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

        JTextPane jtp = new JTextPane();
        StyledDocument doc = (StyledDocument) jtp.getDocument();
        SimpleAttributeSet normal = new SimpleAttributeSet();
        StyleConstants.setFontFamily(normal, "Serif");
        StyleConstants.setFontSize(normal, 72);
        StyleConstants.setForeground(normal, Color.blue);
        doc.insertString(doc.getLength(), "Test", normal);
        jtp.setSelectionStart(doc.getLength());
        jtp.insertIcon(UIManager.getIcon("OptionPane.warningIcon"));
        jtp.setSelectionStart(doc.getLength());
        jtp.insertComponent(new JLabel("Label"));
        jtp.setSelectionStart(doc.getLength());

        ElementIterator iterator = new ElementIterator(doc);
        Element element;
        while ((element = iterator.next()) != null) {
            System.out.println(element);
            AttributeSet as = element.getAttributes();
            if (as.containsAttribute(ELEM, ICON)) {
                System.out.println(StyleConstants.getIcon(as).getClass());
            }
            if (as.containsAttribute(ELEM, COMP)) {
                System.out.println(StyleConstants.getComponent(as).getClass());
            }
        }

        f.add(jtp);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }
}