Java 如何将我的请求焦点事件侦听器设置为线程安全?

Java 如何将我的请求焦点事件侦听器设置为线程安全?,java,multithreading,swing,focus,jtextcomponent,Java,Multithreading,Swing,Focus,Jtextcomponent,我正在使用焦点事件侦听器为所有JTextComponents(JTextField、JTextArea…)实现占位符解决方案。我的实现如下 public class PlaceHolderDecorator { public static void decorate(JTextField field, String placeHoldingText) { field.setForeground(Color.GRAY); field.addFocusListener(new Fo

我正在使用焦点事件侦听器为所有JTextComponents(JTextField、JTextArea…)实现占位符解决方案。我的实现如下

public class PlaceHolderDecorator {

public static void decorate(JTextField field, String placeHoldingText) {
    field.setForeground(Color.GRAY);
    field.addFocusListener(new FocusListener() {
        @Override
        public void focusGained(FocusEvent e) {
            if (field.getText().equals(placeHoldingText)) {
                field.setText("");
                field.setForeground(Color.BLACK);
            }
        }

        @Override
        public void focusLost(FocusEvent e) {
            if (field.getText().isEmpty()) {
                field.setForeground(Color.GRAY);
                field.setText(placeHoldingText);
            }
        }
    });
}
public static void decorate(JTextField field, String placeHoldingText,Color placeHolderColor,Color textColor) {
    field.setForeground(placeHolderColor);
    field.addFocusListener(new FocusListener() {
        @Override
        public void focusGained(FocusEvent e) {
            if (field.getText().equals(placeHoldingText)) {
                field.setText("");
                field.setForeground(textColor);
            }
        }

        @Override
        public void focusLost(FocusEvent e) {
            if (field.getText().isEmpty()) {
                field.setForeground(placeHolderColor);
                field.setText(placeHoldingText);
            }
        }
    });
}

final public static int DATE_FIELD_P = 10;
final public static int PHONE_FIELD_P = 14;

public static void decorate(JFormattedTextField field,String placeHoldingText,Color placeHolderColor,Color textColor,int checkFieldProperty){
    field.setForeground(placeHolderColor);
    final JFormattedTextField.AbstractFormatterFactory formatterFactory = field.getFormatterFactory();
    field.setFormatterFactory(null);
    field.setText(placeHoldingText);
    field.addFocusListener(new FocusListener() {
        @Override
        public void focusGained(FocusEvent e) {
            if (field.getText().equals(placeHoldingText)) {
                field.setText("");
                field.setForeground(textColor);
                field.setFormatterFactory(formatterFactory);
            }
        }

        @Override
        public void focusLost(FocusEvent e) {
            if (field.getText().trim().length()!=checkFieldProperty ) {
                field.setFormatterFactory(null);
                field.setText("");
            }
            if (field.getText().isEmpty()) {
                field.setFormatterFactory(null);
                field.setForeground(placeHolderColor);
                field.setText(placeHoldingText);
            }
        }
    });
 }
}
要使用占位符工具,请执行以下操作

    final Color writingColor = new Color(45, 45, 45);
    final Color holdingColor = new Color(127, 127, 127);

    PlaceHolderDecorator.decorate(jFTFDateDepot, "Date dépot du dossier", holdingColor, writingColor, PlaceHolderDecorator.DATE_FIELD_P);
    PlaceHolderDecorator.decorate(jtfNom, "Nom", holdingColor, writingColor);
这样,当焦点出现时,将显示占位符,在此之后,我需要为我的输入创建一个验证机制,所以我想这样检查我的字段:

    public class RequiredFieldValidator {

        final private List<JTextComponent> components;
        private ErrorDialog dialog;

        public RequiredFieldValidator() {
            components = new ArrayList<>();
        }

        public void add(JTextComponent jTextComponent) {
            components.add(jTextComponent);
        }

        public boolean validate() {

            for (final JTextComponent component : components) {
                String placeHolder = component.getText();
                System.out.println("placeholder : " + placeHolder);

                component.requestFocus();
                if (component.getText().trim().isEmpty()) {
                    System.out.println("validation started");
                    dialog = new ErrorDialog(null, true);
                    dialog.showErrorDialog("Veuillez remplir le champs " + placeHolder + " obligatoir");
                    component.requestFocus();
                    return false;
                }

            }
            return true;
        }
    }
然后对create事件启动验证过程

我在这段代码中遇到的问题是,在运行之前,我希望在create事件上出现类似于序列图的

获取当前组件>>requestFocus()>>fire place holding event>>对组件的文本进行处理>>返回验证>>获取已处理的文本>>如果为空则显示警报>>如果未循环到下一个组件

但是,我惊讶地看到,当前组件上的请求焦点侦听器稍后将在swing线程调度程序上启动


请说明如何使此线程安全,我需要在获取验证文本之前启动焦点侦听器。

您不需要请求组件上的焦点来检查组件是否有文本。当您在所有文本字段中循环进行验证时,焦点一直移动到最后一个文本字段时,用户将不会欣赏它

因此,使用当前的方法,您真正需要的是一种查看文本字段中的文本是否为“placeholder”值的方法。因此,也许您可以将“place horder”逻辑与验证逻辑结合起来,这样两种方法都可以访问place holder值


另一种方法是不将文本字段的文本设置为占位符值。然后验证逻辑可以检查文本字段是否为空。有关使用此方法的解决方案,请查看。它显示提示而不设置文本字段的文本。

使用文本提示是更好的解决方案,适合我的需要。
    fieldValidator = new RequiredFieldValidator();

    fieldValidator.add(jFTFDateDepot);
    fieldValidator.add(jFTFDateNaissance);
    fieldValidator.add(jFTFNumTel);
    fieldValidator.add(jtfLieuNaissance);
    fieldValidator.add(jtfNom);
    fieldValidator.add(jtfPrenom);