Как вывести содержимое в стиле JTextPane в HTML, включая пользовательский стиль?

В настоящее время я использую JTextPane, чтобы пользователи могли добавлять/редактировать текст. Он позволяет выделять полужирным шрифтом/курсивом/подчеркиванием (и я планирую разрешить ссылки в будущем). Это также позволяет пользователям вставлять кнопки, которые вставляются как пользовательские стили. Панель выглядит так:

‹ ‹ изображение удалено › ›

Я хотел бы иметь возможность сохранять/загружать содержимое в формате HTML — содержимое будет включено в Flash SWF. Я могу получить контент в виде HTML следующим образом:

     public String getHTMLText(){

     ByteArrayOutputStream baos = new ByteArrayOutputStream(); 
try{
   HTMLEditorKit hk = new HTMLEditorKit();
   hk.write(baos, this.getStyledDocument(), 0, this.getDocument().getLength());
   
      } catch (IOException e) {
       e.printStackTrace();
      } catch (BadLocationException e) {
       e.printStackTrace();
      }
      return baos.toString();
     }

Это прекрасно работает, если JTextPane включает только полужирный/курсив/подчеркнутый текст. Однако вывод слишком сложен. Я также хочу иметь возможность выводить свой собственный стиль, но когда я пытаюсь, я получаю эту ошибку:

 Exception occurred during event dispatching:
    java.lang.NullPointerException
 at javax.swing.text.html.MinimalHTMLWriter.writeAttributes(MinimalHTMLWriter.java:151)
 at javax.swing.text.html.MinimalHTMLWriter.writeStyles(MinimalHTMLWriter.java:256)
 at javax.swing.text.html.MinimalHTMLWriter.writeHeader(MinimalHTMLWriter.java:220)
 at javax.swing.text.html.MinimalHTMLWriter.write(MinimalHTMLWriter.java:122)
 at javax.swing.text.html.HTMLEditorKit.write(HTMLEditorKit.java:293)
 at javax.swing.text.DefaultEditorKit.write(DefaultEditorKit.java:152)
 at numeracy.referencetextpanel.NRefButtonTextArea.getHTMLText(NRefButtonTextArea.java:328)
 at numeracy.referencetextpanel.NInputPanelRefTextButton.getReferencedText(NInputPanelRefTextButton.java:59)
 at numeracy.referencetextpanel.NInputRefText.actionPerformed(NInputRefText.java:106)
 at javax.swing.AbstractButton.fireActionPerformed(AbstractButton.java:1995)

Мой собственный стиль вставляется так (cID — это строка типа {0-0}):

StyledDocument doc = this.getStyledDocument();

 NRefButton b = this.createRefButton(cID);

 Style style = doc.addStyle(cID, null); //prepare a style
 StyleConstants.setComponent(style, b);  

 doc.insertString(doc.getLength(), b.toString(), style); //insert button at index

Функция createRefButton(String cID):

 private NRefButton createRefButton(String cID) {

  NRefButton b = new NRefButton(_equationButtons.get(cID).getText(), cID, _equationButtons.get(cID).isStruck()); //prepare a button

  return b;
 }

NRefButton переопределяет toString, который возвращает {+cID+}.

Что я хотел бы знать: должен ли я изменить способ вставки стиля, чтобы получить эту ошибку?

Есть ли другой/лучший способ получить HTML из этого JTextPane? Все, что я хочу, это HTML-теги вокруг полужирного/курсивного/подчеркнутого текста, не слишком сложные, как если бы это было так, мне придется удалить ненужный HTML, и чтобы стиль вышел как button.toString().

Или я должен реализовать свой собственный метод toHTML(), оборачивая полужирный/курсив/подчеркнутый текст необходимыми тегами? Я не возражаю против этого (в некотором смысле я бы предпочел это), но я не знаю, как получить стили для данного документа JTextPane. Я полагаю, если бы я мог получить эти стили, я мог бы перебирать их, заключая стилизованный текст в соответствующие теги?

В идеале изображенное содержимое JTextPane будет выводиться как:

<html><p>This is some <b>styled</b> text. It is <u>incredible</u>.
<br/>
<br/>
Here we have a button that has been dropped in: {0-0}. These buttons are a <b><i>required part of this project.</i></b>

Я также хочу иметь возможность читать выходной HTML в JTextPane - опять же, я не против написать для этого свой собственный метод fromHTML(), но мне нужно быть в состоянии получить HTML в первую очередь.

Спасибо, что нашли время, чтобы прочитать мой вопрос.


person Michael Robinson    schedule 11.10.2009    source источник
comment
Я не знаю ответа на ваш вопрос, но, пожалуйста, замените этот пользовательский OutputStream ответом с наибольшим количеством голосов на упомянутый вопрос, т. е. используйте ByteArrayOutputStream.   -  person Dave Ray    schedule 12.10.2009


Ответы (1)


После чтения:

Я написал свой собственный экспортер HTML:

  package com.HTMLExport;

    import javax.swing.text.AbstractDocument;
    import javax.swing.text.AttributeSet;
    import javax.swing.text.BadLocationException;
    import javax.swing.text.Element;
    import javax.swing.text.ElementIterator;
    import javax.swing.text.StyleConstants;
    import javax.swing.text.StyledDocument;

    public class NHTMLWriter {

     private StyledDocument _sd;
     private ElementIterator _it;

     protected static final char NEWLINE = '\n';

     public NHTMLWriter(StyledDocument doc) {
      _sd = doc;
      _it = new ElementIterator(doc.getDefaultRootElement());
     }

     public String getHTML(){
      return "<html>" + this.getBody() + "</html>";
     }

     protected String getBody() {
      /*
           This will be a section element for a styled document.
           We represent this element in HTML as the body tags.
              Therefore we ignore it.
       */
      _it.current();

      Element next = null;

      String body = "<body>";

      while((next = _it.next()) != null) {
       if (this.isText(next)) {
        body += writeContent(next);
       }
       else if(next.getName().equals("component")){
        body += getText(next);  //this is where the custom component is output. 
       }
      }
      body += "</body>";

      return body;
     }
     /**
      * Returns true if the element is a text element.
      */
     protected boolean isText(Element elem) {
      return (elem.getName() == AbstractDocument.ContentElementName);
     }
     protected String writeContent(Element elem){

      AttributeSet attr = elem.getAttributes();

  String startTags = this.getStartTag(attr);

  String content = startTags + this.getText(elem) + this.getEndTag(startTags);

  return content;
     }
     /**
      * Writes out text
      */
     protected String text(Element elem){
      String contentStr = getText(elem);
      if ((contentStr.length() > 0) && (contentStr.charAt(contentStr.length()-1) == NEWLINE)) {
       contentStr = contentStr.substring(0, contentStr.length()-1) + "<br/>";
      }
      if (contentStr.length() > 0) {
       return contentStr;
      }
      return contentStr;
     }

     protected String getText(Element elem){
      try {
       return _sd.getText(elem.getStartOffset(), elem.getEndOffset() - elem.getStartOffset()).replaceAll(NEWLINE+"", "<br/>");
      } catch (BadLocationException e) {
       e.printStackTrace();
      }
      return "";
     }
     private String getEndTag(String startTags) {

  String[] startOrder = startTags.split("<");

  String tags = "";

  for(String s : startOrder){
   tags = "</" + s + tags;
  }

  return tags;
    }
     private String getStartTag(AttributeSet attr) {

      String tag = "";

      if(StyleConstants.isBold(attr)){
       tag += "<b>";
      }
      if(StyleConstants.isItalic(attr)){
       tag += "<i>";
      }
      if(StyleConstants.isUnderline(attr)){
       tag += "<u>";
      }

      return tag;
     }
    }

Теперь мне нужно написать код, чтобы он мог делать обратное: преобразовывать выходной HTML в StyledDocument.

Во-первых, кофе.

person Michael Robinson    schedule 12.10.2009
comment
Хороший. Вы спросили, вы прокомментировали, вы ответили и вы приняли. - person Petah; 04.02.2011
comment
Вы когда-нибудь удосужились написать вторую половину? - person nyxaria; 16.05.2017