JTextPane
ユーザーが入力しているときに特定の単語の色を変更するにはどうすればよいですか?JTextPane
paintComponent
メソッドをオーバーライドする必要がありますか?
5 に答える
いいえ。paintComponent()メソッドをオーバーライドすることは想定されていません。代わりに、を使用する必要がありますStyledDocument
。また、自分で単語を区切る必要があります。
これがデモです。単純なコードエディタのように、入力時に「パブリック」、「保護」、「プライベート」が赤に変わります。
import javax.swing.*;
import java.awt.*;
import javax.swing.text.*;
public class Test extends JFrame {
private int findLastNonWordChar (String text, int index) {
while (--index >= 0) {
if (String.valueOf(text.charAt(index)).matches("\\W")) {
break;
}
}
return index;
}
private int findFirstNonWordChar (String text, int index) {
while (index < text.length()) {
if (String.valueOf(text.charAt(index)).matches("\\W")) {
break;
}
index++;
}
return index;
}
public Test () {
setDefaultCloseOperation(EXIT_ON_CLOSE);
setSize(400, 400);
setLocationRelativeTo(null);
final StyleContext cont = StyleContext.getDefaultStyleContext();
final AttributeSet attr = cont.addAttribute(cont.getEmptySet(), StyleConstants.Foreground, Color.RED);
final AttributeSet attrBlack = cont.addAttribute(cont.getEmptySet(), StyleConstants.Foreground, Color.BLACK);
DefaultStyledDocument doc = new DefaultStyledDocument() {
public void insertString (int offset, String str, AttributeSet a) throws BadLocationException {
super.insertString(offset, str, a);
String text = getText(0, getLength());
int before = findLastNonWordChar(text, offset);
if (before < 0) before = 0;
int after = findFirstNonWordChar(text, offset + str.length());
int wordL = before;
int wordR = before;
while (wordR <= after) {
if (wordR == after || String.valueOf(text.charAt(wordR)).matches("\\W")) {
if (text.substring(wordL, wordR).matches("(\\W)*(private|public|protected)"))
setCharacterAttributes(wordL, wordR - wordL, attr, false);
else
setCharacterAttributes(wordL, wordR - wordL, attrBlack, false);
wordL = wordR;
}
wordR++;
}
}
public void remove (int offs, int len) throws BadLocationException {
super.remove(offs, len);
String text = getText(0, getLength());
int before = findLastNonWordChar(text, offs);
if (before < 0) before = 0;
int after = findFirstNonWordChar(text, offs);
if (text.substring(before, after).matches("(\\W)*(private|public|protected)")) {
setCharacterAttributes(before, after - before, attr, false);
} else {
setCharacterAttributes(before, after - before, attrBlack, false);
}
}
};
JTextPane txt = new JTextPane(doc);
txt.setText("public class Hi {}");
add(new JScrollPane(txt));
setVisible(true);
}
public static void main (String args[]) {
new Test();
}
}
すばやく入力したので、コードはそれほど美しくありませんが、機能します。そして、それがあなたにいくつかのヒントを与えることを願っています。
上書きpaintComponent
しても役に立ちません。
これは簡単なことではありませんが、不可能でもありません。このようなものがあなたを助けます:
DefaultStyledDocument document = new DefaultStyledDocument();
JTextPane textpane = new JTextPane(document);
StyleContext context = new StyleContext();
// build a style
Style style = context.addStyle("test", null);
// set some style properties
StyleConstants.setForeground(style, Color.BLUE);
// add some data to the document
document.insertString(0, "", style);
これを微調整する必要があるかもしれませんが、少なくともどこから始めればよいかがわかります。
別の解決策は、を使用することDocumentFilter
です。
次に例を示します。
DocumentFilterを拡張するクラスを作成します。
private final class CustomDocumentFilter extends DocumentFilter
{
private final StyledDocument styledDocument = yourTextPane.getStyledDocument();
private final StyleContext styleContext = StyleContext.getDefaultStyleContext();
private final AttributeSet greenAttributeSet = styleContext.addAttribute(styleContext.getEmptySet(), StyleConstants.Foreground, Color.GREEN);
private final AttributeSet blackAttributeSet = styleContext.addAttribute(styleContext.getEmptySet(), StyleConstants.Foreground, Color.BLACK);
// Use a regular expression to find the words you are looking for
Pattern pattern = buildPattern();
@Override
public void insertString(FilterBypass fb, int offset, String text, AttributeSet attributeSet) throws BadLocationException {
super.insertString(fb, offset, text, attributeSet);
handleTextChanged();
}
@Override
public void remove(FilterBypass fb, int offset, int length) throws BadLocationException {
super.remove(fb, offset, length);
handleTextChanged();
}
@Override
public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attributeSet) throws BadLocationException {
super.replace(fb, offset, length, text, attributeSet);
handleTextChanged();
}
/**
* Runs your updates later, not during the event notification.
*/
private void handleTextChanged()
{
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
updateTextStyles();
}
});
}
/**
* Build the regular expression that looks for the whole word of each word that you wish to find. The "\\b" is the beginning or end of a word boundary. The "|" is a regex "or" operator.
* @return
*/
private Pattern buildPattern()
{
StringBuilder sb = new StringBuilder();
for (String token : ALL_WORDS_THAT_YOU_WANT_TO_FIND) {
sb.append("\\b"); // Start of word boundary
sb.append(token);
sb.append("\\b|"); // End of word boundary and an or for the next word
}
if (sb.length() > 0) {
sb.deleteCharAt(sb.length() - 1); // Remove the trailing "|"
}
Pattern p = Pattern.compile(sb.toString());
return p;
}
private void updateTextStyles()
{
// Clear existing styles
styledDocument.setCharacterAttributes(0, yourTextPane.getText().length(), blackAttributeSet, true);
// Look for tokens and highlight them
Matcher matcher = pattern.matcher(yourTextPane.getText());
while (matcher.find()) {
// Change the color of recognized tokens
styledDocument.setCharacterAttributes(matcher.start(), matcher.end() - matcher.start(), greenAttributeSet, false);
}
}
}
次に行う必要があるのは、DocumentFilter
作成したJTextPane
を次のようにに適用することだけです。
((AbstractDocument) yourTextPane.getDocument()).setDocumentFilter(new CustomDocumentFilter());
ここでキーワードテキストの色付けを使用して構築しているSQLエディターに対して行ったように、DefaultStyledDocumentを拡張できます...
import java.util.ArrayList;
import java.util.List;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.DefaultStyledDocument;
import javax.swing.text.Style;
public class KeywordStyledDocument extends DefaultStyledDocument {
private static final long serialVersionUID = 1L;
private Style _defaultStyle;
private Style _cwStyle;
public KeywordStyledDocument(Style defaultStyle, Style cwStyle) {
_defaultStyle = defaultStyle;
_cwStyle = cwStyle;
}
public void insertString (int offset, String str, AttributeSet a) throws BadLocationException {
super.insertString(offset, str, a);
refreshDocument();
}
public void remove (int offs, int len) throws BadLocationException {
super.remove(offs, len);
refreshDocument();
}
private synchronized void refreshDocument() throws BadLocationException {
String text = getText(0, getLength());
final List<HiliteWord> list = processWords(text);
setCharacterAttributes(0, text.length(), _defaultStyle, true);
for(HiliteWord word : list) {
int p0 = word._position;
setCharacterAttributes(p0, word._word.length(), _cwStyle, true);
}
}
private static List<HiliteWord> processWords(String content) {
content += " ";
List<HiliteWord> hiliteWords = new ArrayList<HiliteWord>();
int lastWhitespacePosition = 0;
String word = "";
char[] data = content.toCharArray();
for(int index=0; index < data.length; index++) {
char ch = data[index];
if(!(Character.isLetter(ch) || Character.isDigit(ch) || ch == '_')) {
lastWhitespacePosition = index;
if(word.length() > 0) {
if(isReservedWord(word)) {
hiliteWords.add(new HiliteWord(word,(lastWhitespacePosition - word.length())));
}
word="";
}
}
else {
word += ch;
}
}
return hiliteWords;
}
private static final boolean isReservedWord(String word) {
return(word.toUpperCase().trim().equals("CROSS") ||
word.toUpperCase().trim().equals("CURRENT_DATE") ||
word.toUpperCase().trim().equals("CURRENT_TIME") ||
word.toUpperCase().trim().equals("CURRENT_TIMESTAMP") ||
word.toUpperCase().trim().equals("DISTINCT") ||
word.toUpperCase().trim().equals("EXCEPT") ||
word.toUpperCase().trim().equals("EXISTS") ||
word.toUpperCase().trim().equals("FALSE") ||
word.toUpperCase().trim().equals("FETCH") ||
word.toUpperCase().trim().equals("FOR") ||
word.toUpperCase().trim().equals("FROM") ||
word.toUpperCase().trim().equals("FULL") ||
word.toUpperCase().trim().equals("GROUP") ||
word.toUpperCase().trim().equals("HAVING") ||
word.toUpperCase().trim().equals("INNER") ||
word.toUpperCase().trim().equals("INTERSECT") ||
word.toUpperCase().trim().equals("IS") ||
word.toUpperCase().trim().equals("JOIN") ||
word.toUpperCase().trim().equals("LIKE") ||
word.toUpperCase().trim().equals("LIMIT") ||
word.toUpperCase().trim().equals("MINUS") ||
word.toUpperCase().trim().equals("NATURAL") ||
word.toUpperCase().trim().equals("NOT") ||
word.toUpperCase().trim().equals("NULL") ||
word.toUpperCase().trim().equals("OFFSET") ||
word.toUpperCase().trim().equals("ON") ||
word.toUpperCase().trim().equals("ORDER") ||
word.toUpperCase().trim().equals("PRIMARY") ||
word.toUpperCase().trim().equals("ROWNUM") ||
word.toUpperCase().trim().equals("SELECT") ||
word.toUpperCase().trim().equals("SYSDATE") ||
word.toUpperCase().trim().equals("SYSTIME") ||
word.toUpperCase().trim().equals("SYSTIMESTAMP") ||
word.toUpperCase().trim().equals("TODAY") ||
word.toUpperCase().trim().equals("TRUE") ||
word.toUpperCase().trim().equals("UNION") ||
word.toUpperCase().trim().equals("UNIQUE") ||
word.toUpperCase().trim().equals("WHERE"));
}
}
次のようにクラスに追加するだけです。
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Font;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTextPane;
import javax.swing.text.BadLocationException;
import javax.swing.text.Style;
import javax.swing.text.StyleConstants;
import javax.swing.text.StyleContext;
public class SQLEditor extends JFrame {
private static final long serialVersionUID = 1L;
public SQLEditor() {
StyleContext styleContext = new StyleContext();
Style defaultStyle = styleContext.getStyle(StyleContext.DEFAULT_STYLE);
Style cwStyle = styleContext.addStyle("ConstantWidth", null);
StyleConstants.setForeground(cwStyle, Color.BLUE);
StyleConstants.setBold(cwStyle, true);
final JTextPane pane = new JTextPane(new KeywordStyledDocument(defaultStyle, cwStyle));
pane.setFont(new Font("Courier New", Font.PLAIN, 12));
JScrollPane scrollPane = new JScrollPane(pane);
getContentPane().add(scrollPane, BorderLayout.CENTER);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setSize(375, 400);
}
public static void main(String[] args) throws BadLocationException {
SQLEditor app = new SQLEditor();
app.setVisible(true);
}
}
これが欠落しているHiliteWordクラスです...
public class HiliteWord {
int _position;
String _word;
public HiliteWord(String word, int position) {
_position = position;
_word = word;
}
}
@コンスタンティン
親愛なるコンスタンティン、私は私の小さなプロジェクトにあなたの素晴らしいソリューションを使用しました、そしてほんの少しの調整の後、あなたのソリューションは私のためにうまくいきました。
許可すれば、私の変更は次のとおりです。
自分のJFrameでのクラスKeywordStyledDocumentの使用:
StyleContext styleContext = new StyleContext();
Style defaultStyle = styleContext.getStyle(StyleContext.DEFAULT_STYLE);
私が変更したこの行:MutableAttributeSet cwStyle = Functions.style(true、false、Color.RED);
private JTextPane jTextPaneNumbers = new JTextPane(new KeywordStyledDocument(defaultStyle, cwStyle));
styleと呼ばれる静的関数でcwStyleインスタンスの供給を外部委託しました。
public static MutableAttributeSet style(boolean boldness, boolean italic, Color color) {
MutableAttributeSet s = new SimpleAttributeSet();
StyleConstants.setLineSpacing(s, -0.2f);
StyleConstants.setBold(s, boldness);
StyleConstants.setItalic(s, italic);
StyleConstants.setForeground(s, color);
return s;
}
さらに、上記のように、cwStyleクラスはStyleConstantsのインスタンスではなく、MutableAttributeSetのインスタンスです。したがって、当然、KeywordStyledDocumentClassのコンストラクターも変更する必要がありました。
public KeywordStyledDocument(Style defaultStyle, MutableAttributeSet cwStyle) {
_defaultStyle = defaultStyle;
_cwStyle = cwStyle;
}
このlitleが変更され、isReservedWord関数に自分の「単語」が追加された後、自分の文字'と*をprocessWord関数に追加します。
...word.toUpperCase().trim().equals("UNION") ||
word.toUpperCase().trim().equals("UNIQUE") ||
word.toUpperCase().trim().equals("WHERE") ||
word.trim().equals("''''''") ||
word.trim().equals("******")
);
if(!(Character.isLetter(ch) || Character.isDigit(ch) || ch == '_' || ch == '\'' || ch == '*')) {
私は私の望みの結果になりました:
ここにコードを表示していただき、ありがとうございます。