【问题标题】:Autocomplete jtextfield from specific length从特定长度自动完成 jtextfield
【发布时间】:2016-06-14 20:02:07
【问题描述】:

我想执行 jtextfield 的自动完成,为此我在谷歌上搜索并找到了一些很好的示例和代码 sn-ps,但我的场景与所有这些不同,通常每个示例都会自动完成字段,例如比如说,如果我按 S 它将自动完成 Stackoverflow 或者如果我按 G 它将完成 google等,意味着它只需要第一个字符即可自动完成。

但我的问题是,我想执行自动完成 当前 4 个字符 将输入或输入时说如果我输入 stac 应该是自动完成它到 stackoverflow,而不是第一个字符 S。

请指导我如何做到这一点,我在这个问题上陷入了困境。

以下代码用于我的工作类

package testautocomp;

import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import javax.swing.JTextField;
import javax.swing.SwingUtilities;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import javax.swing.text.BadLocationException;

public class JTextFieldAutoCompletion extends JTextField implements DocumentListener, ActionListener{

    /**
     * 
     */
    private static final long serialVersionUID = 4810213451949301347L;

    //Les Données De L'AutoCompletion
    private List<String> data = new ArrayList<String>();

    //Un Constructeur Par Défaut
    public JTextFieldAutoCompletion() {
        //Par Defaut Le Nombre de caractère visible dans le champs de texte est 25
        this(25);
    }
    /**
     * Un Constructeur Paramétré
     * @param columns nombre de caractère visible dans le champs de texte
     */
    public JTextFieldAutoCompletion(int columns) {
        //passer au constructeur à deux arguments le nombre de colonne visible dans le champs de texte et definir les données de l'autocompletion à null.
        this(columns, null);
    }
    /**
     * Constructeur Paramétré à deux arguments
     * @param columns nombre de caractère visible dans le champs de texte
     * @param data les données de l'autocompletion
     */
    public JTextFieldAutoCompletion(int columns, List<String> data) {
        super(columns);
        //ici on fait appel à la méthode setDataCompletion pour definir les données de l'autocompletion
        this.setDataCompletion(data);
        //je défini l'ecouteur de l'evenement de la saisie
        this.getDocument().addDocumentListener(this);
        //je défini j'ecouteur de la touche entrer
        this.addActionListener(this);
    }
    /**
     * Permet De Redefinir les données de l'autocompletion
     * @param data les données de l'autocompletion
     */
    public void setDataCompletion(List<String> data) {
        //on affecte seulement si data est déffirent à null
        if(data != null)
            this.data = data;
        //on va trier les données de l'autocompletion 
        Collections.sort(this.data);
    }
    /**
     * Evenement Déclenché à chaque fois que l'utilisateur tape un caractère quelconque, ou fasse une copier/coller dans le champs de texte.
     */
    @Override
    public void insertUpdate(DocumentEvent e) {
        // TODO Stub de la méthode généré automatiquement
        //on arréte l'exécution de l'evenement si l'utilisateur fasse une copier/coller
        if(e.getLength() != 1) return;

        //on récupére la position du dernier caratère saisie en comptant de zéro, premier caractère est en position 0, le deuxième à 2 etc..
        int pos = e.getOffset();
        String prefix = null;
        try {
            //on recupére dans prefix ce qu'a saisi l'utilisateur jusqu'à présent.
            prefix = this.getText(0, pos + 1);
        } catch (BadLocationException e1) {}

        //on fait une recherche sur la chaine qu'a saisi l'utilisateur dans les données de l'autocompletion. 
        //la méthode binarySearch retourne :
        //Soit l'index de l'element cherché s'il est contenu dans la collection.
        //Soit le nombre d'element de la collection si tous les elements sont inférieurs à l'element qu'on cherche.
        //Soit un entier négatif qui représente l'index de premier element supérieur de l'element qu'on cherche.
        int index = Collections.binarySearch(data, prefix);

        if(index < 0 && -index <= data.size()) {
            //Completion Trouvé
            //On récupére le premier element supérieur à l'element cherché. le signe - retourne la valeur absolue de la variable index. 
            String match = data.get(-index - 1);

            //on s'assure que la chaine dans la variable match commence par la chaine contenu dans la variable prefix c-à-d ce qu'a saisi l'utilisateur 
            if(match.startsWith(prefix)) {
                //si oui on met on place l'autocompletion sinon on fait rien :).
                SwingUtilities.invokeLater(new AutoCompletion(pos, match));
            }
        } else ;
            //Aucune Completion Trouvé

    }
    /**
     * Permet De Valider L'AutoCompletion En Cliquant Sur La Touche Entrer
     */
    @Override
    public void actionPerformed(ActionEvent e) {
        // TODO Stub de la méthode généré automatiquement
        setCaretPosition(getSelectionEnd());
    }
    @Override
    public void removeUpdate(DocumentEvent e) {}
    @Override
    public void changedUpdate(DocumentEvent e) {}

    private class AutoCompletion implements Runnable{
        private int pos;
        private String completion;

        public AutoCompletion(int pos, String completion) {
            this.pos = pos;
            this.completion = completion;
        }
        @Override
        public void run() {
            // TODO Stub de la méthode généré automatiquement
            //On affecte la chaine trouvé pour l'autocompletion dans le champs de texte
            setText(completion);
            //on definit à partir d'où va débuter la séléction des caractères ajouté comme completion. 
            //j'ai précisé qu'il va débuter de la fin vers le dernier caractère sasie par l'utilisateur
            setCaretPosition(completion.length());
            //j'ai appliqué la séléction jusqu'au dernier caractère sasie par l'utilisateur
            moveCaretPosition(pos + 1);
        }       
    }
}

这是我的主要课程

package testautocomp;

import java.util.ArrayList;
import java.util.List;

import javax.swing.JFrame;

public class TestAutoComp {
    public static void main(String[] a) {
        JFrame f = new JFrame();
        JTextFieldAutoCompletion tf = new JTextFieldAutoCompletion(25);

        List<String> data = new ArrayList<String>();
        data.add("jtextfield autocompletion");
        data.add("java swing");
        data.add("je veux apprendre java");
        data.add("stackoverflow");
        data.add("ahad");       


        tf.setDataCompletion(data);
        f.getContentPane().add(tf, "North");

        f.pack();f.setDefaultCloseOperation(f.EXIT_ON_CLOSE);
        f.setLocationRelativeTo(null); 
        f.setVisible(true);
    } 
}

【问题讨论】:

  • (1-) i google it and found some good examples - 然后修改示例以满足您的要求。您可以轻松检查当前在文本字段中输入的字符数。
  • @(1-) 我在上面发布了我的代码,请您帮我解决这个问题。
  • 我确实提供了帮助。您检查文本字段中文本的长度。如果长度是您想要的,则进行处理,否则您只需退出该方法。

标签: java swing swingx


【解决方案1】:

根据您的问题,您希望在输入前 4 个字符时执行自动完成,然后只需检查您的 insertUpdate() 方法,文本字段中的文本长度大于 3,因此会得到预期输出。 我已经修改了您的代码以获得预期的结果。

 package temp;
 import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import javax.swing.JTextField;
import javax.swing.SwingUtilities;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import javax.swing.text.BadLocationException;

public class JTextFieldAutoCompletion extends JTextField implements DocumentListener, ActionListener{

/**
 * 
 */
private static final long serialVersionUID = 4810213451949301347L;

//Les Données De L'AutoCompletion
private List<String> data = new ArrayList<String>();

//Un Constructeur Par Défaut
public JTextFieldAutoCompletion() {
    //Par Defaut Le Nombre de caractère visible dans le champs de texte est 25
    this(25);
}
/**
 * Un Constructeur Paramétré
 * @param columns nombre de caractère visible dans le champs de texte
 */
public JTextFieldAutoCompletion(int columns) {
    //passer au constructeur à deux arguments le nombre de colonne visible dans le champs de texte et definir les données de l'autocompletion à null.
    this(columns, null);
}
/**
 * Constructeur Paramétré à deux arguments
 * @param columns nombre de caractère visible dans le champs de texte
 * @param data les données de l'autocompletion
 */
public JTextFieldAutoCompletion(int columns, List<String> data) {
    super(columns);
    //ici on fait appel à la méthode setDataCompletion pour definir les données de l'autocompletion
    this.setDataCompletion(data);
    //je défini l'ecouteur de l'evenement de la saisie
    this.getDocument().addDocumentListener(this);
    //je défini j'ecouteur de la touche entrer
    this.addActionListener(this);
}
/**
 * Permet De Redefinir les données de l'autocompletion
 * @param data les données de l'autocompletion
 */
public void setDataCompletion(List<String> data) {
    //on affecte seulement si data est déffirent à null
    if(data != null)
        this.data = data;
    //on va trier les données de l'autocompletion 
    Collections.sort(this.data);
}
/**
 * Evenement Déclenché à chaque fois que l'utilisateur tape un caractère quelconque, ou fasse une copier/coller dans le champs de texte.
 */
@Override
public void insertUpdate(DocumentEvent e) {
    // TODO Stub de la méthode généré automatiquement
    //on arréte l'exécution de l'evenement si l'utilisateur fasse une copier/coller

    if(this.getText().length()>3){
    if(e.getLength() != 1) return;

    //on récupére la position du dernier caratère saisie en comptant de zéro, premier caractère est en position 0, le deuxième à 2 etc..
    int pos = e.getOffset();
    String prefix = null;
    try {
        //on recupére dans prefix ce qu'a saisi l'utilisateur jusqu'à présent.
        prefix = this.getText(0, pos + 1);
    } catch (BadLocationException e1) {}

    //on fait une recherche sur la chaine qu'a saisi l'utilisateur dans les données de l'autocompletion. 
    //la méthode binarySearch retourne :
    //Soit l'index de l'element cherché s'il est contenu dans la collection.
    //Soit le nombre d'element de la collection si tous les elements sont inférieurs à l'element qu'on cherche.
    //Soit un entier négatif qui représente l'index de premier element supérieur de l'element qu'on cherche.
    int index = Collections.binarySearch(data, prefix);

    if(index < 0 && -index <= data.size()) {
        //Completion Trouvé
        //On récupére le premier element supérieur à l'element cherché. le signe - retourne la valeur absolue de la variable index. 
        String match = data.get(-index - 1);

        //on s'assure que la chaine dans la variable match commence par la chaine contenu dans la variable prefix c-à-d ce qu'a saisi l'utilisateur 
        if(match.startsWith(prefix)) {
            //si oui on met on place l'autocompletion sinon on fait rien :).
            SwingUtilities.invokeLater(new AutoCompletion(pos, match));
        }
    } else ;
        //Aucune Completion Trouvé
    }
}
/**
 * Permet De Valider L'AutoCompletion En Cliquant Sur La Touche Entrer
 */
@Override
public void actionPerformed(ActionEvent e) {
    // TODO Stub de la méthode généré automatiquement
    setCaretPosition(getSelectionEnd());
}
@Override
public void removeUpdate(DocumentEvent e) {}
@Override
public void changedUpdate(DocumentEvent e) {}

private class AutoCompletion implements Runnable{
    private int pos;
    private String completion;

    public AutoCompletion(int pos, String completion) {
        this.pos = pos;
        this.completion = completion;
    }
    @Override
    public void run() {
        // TODO Stub de la méthode généré automatiquement
        //On affecte la chaine trouvé pour l'autocompletion dans le champs de texte
        setText(completion);
        //on definit à partir d'où va débuter la séléction des caractères ajouté comme completion. 
        //j'ai précisé qu'il va débuter de la fin vers le dernier caractère sasie par l'utilisateur
        setCaretPosition(completion.length());
        //j'ai appliqué la séléction jusqu'au dernier caractère sasie par l'utilisateur
        moveCaretPosition(pos + 1);
    }       
}

}

【讨论】:

    猜你喜欢
    • 2012-01-31
    • 2012-08-09
    • 2012-06-06
    • 2013-03-15
    • 2012-02-15
    • 2012-05-13
    • 1970-01-01
    • 1970-01-01
    • 2013-01-01
    相关资源
    最近更新 更多